home *** CD-ROM | disk | FTP | other *** search
/ Chip 2007 January, February, March & April / Chip-Cover-CD-2007-02.iso / Pakiet internetowy / Przegladarki internetowe / Mozilla Seamonkey 1.0.5 pl / seamonkey-1.0.5.pl-PL.win32.installer.exe / CHATZILLA.XPI / bin / chrome / chatzilla.jar / content / chatzilla / static.js < prev   
Encoding:
Text File  |  2005-07-27  |  121.2 KB  |  4,211 lines

  1. /* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*-
  2.  *
  3.  * ***** BEGIN LICENSE BLOCK *****
  4.  * Version: MPL 1.1/GPL 2.0/LGPL 2.1
  5.  *
  6.  * The contents of this file are subject to the Mozilla Public License Version
  7.  * 1.1 (the "License"); you may not use this file except in compliance with
  8.  * the License. You may obtain a copy of the License at
  9.  * http://www.mozilla.org/MPL/
  10.  *
  11.  * Software distributed under the License is distributed on an "AS IS" basis,
  12.  * WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
  13.  * for the specific language governing rights and limitations under the
  14.  * License.
  15.  *
  16.  * The Original Code is ChatZilla.
  17.  *
  18.  * The Initial Developer of the Original Code is
  19.  * Netscape Communications Corporation.
  20.  * Portions created by the Initial Developer are Copyright (C) 1998
  21.  * the Initial Developer. All Rights Reserved.
  22.  *
  23.  * Contributor(s):
  24.  *   Robert Ginda, <rginda@netscape.com>, original author
  25.  *   Chiaki Koufugata chiaki@mozilla.gr.jp UI i18n
  26.  *   Samuel Sieb, samuel@sieb.net, MIRC color codes, munger menu, and various
  27.  *
  28.  * Alternatively, the contents of this file may be used under the terms of
  29.  * either the GNU General Public License Version 2 or later (the "GPL"), or
  30.  * the GNU Lesser General Public License Version 2.1 or later (the "LGPL"),
  31.  * in which case the provisions of the GPL or the LGPL are applicable instead
  32.  * of those above. If you wish to allow use of your version of this file only
  33.  * under the terms of either the GPL or the LGPL, and not to allow others to
  34.  * use your version of this file under the terms of the MPL, indicate your
  35.  * decision by deleting the provisions above and replace them with the notice
  36.  * and other provisions required by the GPL or the LGPL. If you do not delete
  37.  * the provisions above, a recipient may use your version of this file under
  38.  * the terms of any one of the MPL, the GPL or the LGPL.
  39.  *
  40.  * ***** END LICENSE BLOCK ***** */
  41.  
  42. const __cz_version   = "0.9.67+";
  43. const __cz_condition = "green";
  44. const __cz_suffix    = "";
  45. const __cz_guid      = "59c81df5-4b7a-477b-912d-4e0fdf64e5f2";
  46.  
  47. var warn;
  48. var ASSERT;
  49. var TEST;
  50.  
  51. if (DEBUG)
  52. {
  53.     _dd_pfx = "cz: ";
  54.     warn = function (msg) { dumpln ("** WARNING " + msg + " **"); }
  55.     TEST = ASSERT = function _assert(expr, msg) {
  56.                  if (!expr) {
  57.                      dd("** ASSERTION FAILED: " + msg + " **\n" +
  58.                         getStackTrace() + "\n");
  59.                      return false;
  60.                  } else {
  61.                      return true;
  62.                  }
  63.              }
  64. }
  65. else
  66.     dd = warn = TEST = ASSERT = function (){};
  67.  
  68. var client = new Object();
  69.  
  70. client.TYPE = "IRCClient";
  71. client.COMMAND_CHAR = "/";
  72. client.STEP_TIMEOUT = 100;
  73. client.MAX_MESSAGES = 200;
  74. client.MAX_HISTORY = 50;
  75. /* longest nick to show in display before forcing the message to a block level
  76.  * element */
  77. client.MAX_NICK_DISPLAY = 14;
  78. /* longest word to show in display before abbreviating */
  79. client.MAX_WORD_DISPLAY = 20;
  80. client.PRINT_DIRECTION = 1; /*1 => new messages at bottom, -1 => at top */
  81.  
  82. client.MAX_MSG_PER_ROW = 3; /* default number of messages to collapse into a
  83.                              * single row, max. */
  84. client.INITIAL_COLSPAN = 5; /* MAX_MSG_PER_ROW cannot grow to greater than
  85.                              * one half INITIAL_COLSPAN + 1. */
  86. client.NOTIFY_TIMEOUT = 5 * 60 * 1000; /* update notify list every 5 minutes */
  87. client.AWAY_TIMEOUT = 2 * 60 * 1000; /* update away status every 2 mins */
  88.  
  89. client.SLOPPY_NETWORKS = true; /* true if msgs from a network can be displayed
  90.                                 * on the current object if it is related to
  91.                                 * the network (ie, /whois results will appear
  92.                                 * on the channel you're viewing, if that channel
  93.                                 * is on the network that the results came from)
  94.                                 */
  95. client.DOUBLETAB_TIME = 500;
  96. client.IMAGEDIR = "chrome://chatzilla/skin/images/";
  97. client.HIDE_CODES = true;      /* true if you'd prefer to show numeric response
  98.                                 * codes as some default value (ie, "===") */
  99. /* true if the browser widget shouldn't be allowed to take focus.  windows, and
  100.  * probably the mac, need to be able to give focus to the browser widget for
  101.  * copy to work properly. */
  102. client.NO_BROWSER_FOCUS = (navigator.platform.search(/mac|win/i) == -1);
  103. client.DEFAULT_RESPONSE_CODE = "===";
  104.  
  105.  
  106. client.viewsArray = new Array();
  107. client.activityList = new Object();
  108. client.inputHistory = new Array();
  109. client.lastHistoryReferenced = -1;
  110. client.incompleteLine = "";
  111. client.lastTabUp = new Date();
  112.  
  113. CIRCNetwork.prototype.INITIAL_CHANNEL = "";
  114. CIRCNetwork.prototype.MAX_MESSAGES = 100;
  115. CIRCNetwork.prototype.IGNORE_MOTD = false;
  116. CIRCNetwork.prototype.RECLAIM_WAIT = 15000;
  117. CIRCNetwork.prototype.RECLAIM_TIMEOUT = 400000;
  118.  
  119. CIRCServer.prototype.READ_TIMEOUT = 0;
  120. CIRCServer.prototype.PRUNE_OLD_USERS = 0; // prune on user quit.
  121.  
  122. CIRCUser.prototype.MAX_MESSAGES = 200;
  123.  
  124. CIRCChannel.prototype.MAX_MESSAGES = 300;
  125.  
  126. CIRCChanUser.prototype.MAX_MESSAGES = 200;
  127.  
  128. function init()
  129. {
  130.     if (("initialized" in client) && client.initialized)
  131.         return;
  132.  
  133.     client.initialized = false;
  134.  
  135.     client.networks = new Object();
  136.     client.entities = new Object();
  137.     client.eventPump = new CEventPump (200);
  138.  
  139.     if (DEBUG)
  140.     {
  141.         /* hook all events EXCEPT server.poll and *.event-end types
  142.          * (the 4th param inverts the match) */
  143.         client.debugHook =
  144.             client.eventPump.addHook([{type: "poll", set:/^(server|dcc-chat)$/},
  145.                                     {type: "event-end"}], event_tracer,
  146.                                     "event-tracer", true /* negate */,
  147.                                     false /* disable */);
  148.     }
  149.  
  150.     initApplicationCompatibility();
  151.     initMessages();
  152.     if (client.host == "")
  153.         showErrorDlg(getMsg(MSG_ERR_UNKNOWN_HOST, getBrowserURL()));
  154.  
  155.     initRDF();
  156.     initCommands();
  157.     initPrefs();
  158.     initMunger();
  159.     initNetworks();
  160.     initMenus();
  161.     initStatic();
  162.     initHandlers();
  163.  
  164.     // Create DCC handler.
  165.     client.dcc = new CIRCDCC(client);
  166.  
  167.     // start logging.  nothing should call display() before this point.
  168.     if (client.prefs["log"])
  169.         client.openLogFile(client);
  170.  
  171.     client.display(MSG_WELCOME, "HELLO");
  172.     client.dispatch("set-current-view", { view: client });
  173.  
  174.     importFromFrame("updateHeader");
  175.     importFromFrame("setHeaderState");
  176.     importFromFrame("changeCSS");
  177.     importFromFrame("addUsers");
  178.     importFromFrame("updateUsers");
  179.     importFromFrame("removeUsers");
  180.  
  181.     processStartupScripts();
  182.  
  183.     client.commandManager.installKeys(document);
  184.     createMenus();
  185.  
  186.     initIcons();
  187.  
  188.     client.busy = false;
  189.     updateProgress();
  190.  
  191.     client.initialized = true;
  192.  
  193.     dispatch("networks");
  194.     dispatch("commands");
  195.  
  196.     processStartupURLs();
  197. }
  198.  
  199. function initStatic()
  200. {
  201.     client.mainWindow = window;
  202.  
  203.     try
  204.     {
  205.         var io = Components.classes['@mozilla.org/network/io-service;1'];
  206.         client.iosvc = io.getService(Components.interfaces.nsIIOService);
  207.     }
  208.     catch (ex)
  209.     {
  210.         dd("IO service failed to initalize: " + ex);
  211.     }
  212.  
  213.     try
  214.     {
  215.         const nsISound = Components.interfaces.nsISound;
  216.         client.sound =
  217.             Components.classes["@mozilla.org/sound;1"].createInstance(nsISound);
  218.  
  219.         client.soundList = new Object();
  220.     }
  221.     catch (ex)
  222.     {
  223.         dd("Sound failed to initalize: " + ex);
  224.     }
  225.  
  226.     try
  227.     {
  228.         const nsIGlobalHistory = Components.interfaces.nsIGlobalHistory;
  229.         const GHIST_CONTRACTID = "@mozilla.org/browser/global-history;1";
  230.         client.globalHistory =
  231.             Components.classes[GHIST_CONTRACTID].getService(nsIGlobalHistory);
  232.     }
  233.     catch (ex)
  234.     {
  235.         dd("Global History failed to initalize: " + ex);
  236.     }
  237.  
  238.     try
  239.     {
  240.         const nsISDateFormat = Components.interfaces.nsIScriptableDateFormat;
  241.         const DTFMT_CID = "@mozilla.org/intl/scriptabledateformat;1";
  242.         client.dtFormatter =
  243.             Components.classes[DTFMT_CID].createInstance(nsISDateFormat);
  244.  
  245.         // Mmmm, fun. This ONLY affects the ChatZilla window, don't worry!
  246.         Date.prototype.toStringInt = Date.prototype.toString;
  247.         Date.prototype.toString = function() {
  248.             var dtf = client.dtFormatter;
  249.             return dtf.FormatDateTime("", dtf.dateFormatLong,
  250.                                       dtf.timeFormatSeconds,
  251.                                       this.getFullYear(), this.getMonth() + 1,
  252.                                       this.getDate(), this.getHours(),
  253.                                       this.getMinutes(), this.getSeconds()
  254.                                      );
  255.         }
  256.     }
  257.     catch (ex)
  258.     {
  259.         dd("Locale-correct date formatting failed to initalize: " + ex);
  260.     }
  261.  
  262.     multilineInputMode(client.prefs["multiline"]);
  263.     if (client.prefs["showModeSymbols"])
  264.         setListMode("symbol");
  265.     else
  266.         setListMode("graphic");
  267.     setDebugMode(client.prefs["debugMode"]);
  268.  
  269.     var ver = __cz_version + (__cz_suffix ? "-" + __cz_suffix : "");
  270.  
  271.     var ua = navigator.userAgent;
  272.     var app = getService("@mozilla.org/xre/app-info;1", "nsIXULAppInfo");
  273.     if (app)
  274.     {
  275.         // Use the XUL host app info, and Gecko build ID.
  276.         if (app.ID == "{" + __cz_guid + "}")
  277.         {
  278.             // We ARE the app, in other words, we're running in XULrunner.
  279.             // Because of this, we must disregard app.(name|vendor|version).
  280.             // "XULRunner 1.7+/2005071506"
  281.             ua = "XULRunner " + app.platformVersion + "/" + app.platformBuildID;
  282.  
  283.             // "XULRunner 1.7+/2005071506, Windows"
  284.             CIRCServer.prototype.HOST_RPLY = ua + ", " + client.platform;
  285.         }
  286.         else
  287.         {
  288.             // "Firefox 1.0+/2005071506"
  289.             ua = app.name + " " + app.version + "/";
  290.             if ("platformBuildID" in app) // 1.1 and up
  291.                 ua += app.platformBuildID;
  292.             else if ("geckoBuildID" in app) // 1.0 - 1.1 trunk only
  293.                 ua += app.geckoBuildID;
  294.             else // Uh oh!
  295.                 ua += "??????????";
  296.  
  297.             // "Mozilla Firefox 1.0+, Windows"
  298.             CIRCServer.prototype.HOST_RPLY = app.vendor + " " + app.name + " " +
  299.                                              app.version + ", " + client.platform;
  300.         }
  301.     }
  302.     else
  303.     {
  304.         // Extract the revision number, and Gecko build ID.
  305.         var ary = navigator.userAgent.match(/(rv:[^;)\s]+).*?Gecko\/(\d+)/);
  306.         if (ary)
  307.         {
  308.             if (navigator.vendor)
  309.                 ua = navigator.vendor + " " + navigator.vendorSub; // FF 1.0
  310.             else
  311.                 ua = client.entities.brandShortName + " " + ary[1]; // Suite
  312.             ua = ua + "/" + ary[2];
  313.         }
  314.         CIRCServer.prototype.HOST_RPLY = client.entities.brandShortName + ", " +
  315.                                          client.platform;
  316.     }
  317.  
  318.     client.userAgent = getMsg(MSG_VERSION_REPLY, [ver, ua]);
  319.     CIRCServer.prototype.VERSION_RPLY = client.userAgent;
  320.     CIRCServer.prototype.SOURCE_RPLY = MSG_SOURCE_REPLY;
  321.  
  322.     client.statusBar = new Object();
  323.  
  324.     client.statusBar["server-nick"] = document.getElementById ("server-nick");
  325.  
  326.     client.statusElement = document.getElementById("status-text");
  327.     client.defaultStatus = MSG_DEFAULT_STATUS;
  328.  
  329.     client.progressPanel = document.getElementById("status-progress-panel");
  330.     client.progressBar = document.getElementById("status-progress-bar");
  331.  
  332.     client.logFile = null;
  333.     setInterval("onNotifyTimeout()", client.NOTIFY_TIMEOUT);
  334.     setInterval("onWhoTimeout()", client.AWAY_TIMEOUT);
  335.  
  336.     client.defaultCompletion = client.COMMAND_CHAR + "help ";
  337.  
  338.     client.deck = document.getElementById('output-deck');
  339. }
  340.  
  341. function initApplicationCompatibility()
  342. {
  343.     // This routine does nothing more than tweak the UI based on the host
  344.     // application.
  345.  
  346.     // Set up simple host and platform information.
  347.     client.host = "Unknown";
  348.     if ("getBrowserURL" in window)
  349.     {
  350.         var url = getBrowserURL();
  351.         if (url == "chrome://navigator/content/navigator.xul")
  352.             client.host = "Mozilla";
  353.         else if (url == "chrome://browser/content/browser.xul")
  354.             client.host = "Firefox";
  355.         else
  356.             client.host = ""; // We don't know this host. Show an error later.
  357.     }
  358.     else
  359.     {
  360.         client.host = "XULrunner";
  361.     }
  362.  
  363.     client.platform = "Unknown";
  364.     if (navigator.platform.search(/mac/i) > -1)
  365.         client.platform = "Mac";
  366.     if (navigator.platform.search(/win/i) > -1)
  367.         client.platform = "Windows";
  368.     if (navigator.platform.search(/linux/i) > -1)
  369.         client.platform = "Linux";
  370.     if (navigator.platform.search(/os\/2/i) > -1)
  371.         client.platform = "OS/2";
  372.  
  373.     client.hostPlatform = client.host + client.platform;
  374.  
  375.     CIRCServer.prototype.OS_RPLY = navigator.oscpu + " (" +
  376.                                    navigator.platform + ")";
  377.  
  378.     // Windows likes \r\n line endings, as wussy-notepad can't cope with just
  379.     // \n logs.
  380.     if (client.platform == "Windows")
  381.         client.lineEnd = "\r\n";
  382.     else
  383.         client.lineEnd = "\n";
  384. }
  385.  
  386. function initNetworks()
  387. {
  388.     client.addNetwork("moznet",
  389.                       [{name: "irc.mozilla.org", port:6667}, 
  390.                        {name: "irc.mozilla.org", port:6697, isSecure:true}]);
  391.     client.addNetwork("hybridnet", [{name: "irc.ssc.net", port: 6667}]);
  392.     client.addNetwork("slashnet", [{name: "irc.slashnet.org", port:6667}]);
  393.     client.addNetwork("dalnet", [{name: "irc.dal.net", port:6667}]);
  394.     client.addNetwork("undernet", [{name: "irc.undernet.org", port:6667}]);
  395.     client.addNetwork("webbnet", [{name: "irc.webbnet.info", port:6667}]);
  396.     client.addNetwork("quakenet", [{name: "irc.quakenet.org", port:6667}]);
  397.     client.addNetwork("freenode", [{name: "irc.freenode.net", port:6667}]);
  398.     client.addNetwork("serenia", 
  399.                       [{name: "chat.serenia.net", port:9999, isSecure:true}]);
  400.     client.addNetwork("efnet",
  401.                       [{name: "irc.prison.net", port: 6667},
  402.                        {name: "irc.magic.ca", port: 6667}]);
  403. }
  404.  
  405. function initIcons()
  406. {
  407.     // Make sure we got the ChatZilla icon(s) in place first.
  408.     const iconName = "chatzilla-window";
  409.     const suffixes = [".ico", ".xpm", "16.xpm"];
  410.  
  411.     /* when installing on Mozilla, the XPI has the power to put the icons where
  412.      * they are needed - in Firefox, it doesn't. So we move them here, instead.
  413.      */
  414.     if (client.host != "Firefox")
  415.         return;
  416.  
  417.     var sourceDir = getSpecialDirectory("ProfD");
  418.     sourceDir.append("extensions");
  419.     sourceDir.append("{" + __cz_guid + "}");
  420.     sourceDir.append("defaults");
  421.  
  422.     var destDir = getSpecialDirectory("AChrom");
  423.     destDir.append("icons");
  424.     destDir.append("default");
  425.     if (!destDir.exists())
  426.     {
  427.         try
  428.         {
  429.             mkdir(destDir);
  430.         }
  431.         catch(ex)
  432.         {
  433.             return;
  434.         }
  435.     }
  436.  
  437.     for (var i = 0; i < suffixes.length; i++)
  438.     {
  439.         var iconDest = destDir.clone();
  440.         iconDest.append(iconName + suffixes[i]);
  441.         var iconSrc = sourceDir.clone();
  442.         iconSrc.append(iconName + suffixes[i]);
  443.  
  444.         if (iconSrc.exists() && !iconDest.exists())
  445.         {
  446.             try
  447.             {
  448.                 iconSrc.copyTo(iconDest.parent, iconDest.leafName);
  449.             }
  450.             catch(ex){}
  451.         }
  452.     }
  453. }
  454.  
  455. function getFindData(e)
  456. {
  457.     var findData = new nsFindInstData();
  458.     findData.browser = e.sourceObject.frame;
  459.     findData.rootSearchWindow = e.sourceObject.frame.contentWindow;
  460.     findData.currentSearchWindow = e.sourceObject.frame.contentWindow;
  461.     return findData;
  462. }
  463.  
  464. function importFromFrame(method)
  465. {
  466.     client.__defineGetter__(method, import_wrapper);
  467.     CIRCNetwork.prototype.__defineGetter__(method, import_wrapper);
  468.     CIRCChannel.prototype.__defineGetter__(method, import_wrapper);
  469.     CIRCUser.prototype.__defineGetter__(method, import_wrapper);
  470.     CIRCDCCChat.prototype.__defineGetter__(method, import_wrapper);
  471.     CIRCDCCFileTransfer.prototype.__defineGetter__(method, import_wrapper);
  472.  
  473.     function import_wrapper()
  474.     {
  475.         var dummy = function(){};
  476.  
  477.         if (!("frame" in this))
  478.             return dummy;
  479.  
  480.         try
  481.         {
  482.             var window = getContentWindow(this.frame)
  483.             if (window && "initialized" in window && window.initialized &&
  484.                 method in window)
  485.             {
  486.                 return window[method];
  487.             }
  488.         }
  489.         catch (ex)
  490.         {
  491.             ASSERT(0, "Caught exception calling: " + method + "\n" + ex);
  492.         }
  493.  
  494.         return dummy;
  495.     };
  496. }
  497.  
  498. function processStartupScripts()
  499. {
  500.     client.plugins = new Array();
  501.     var scripts = client.prefs["initialScripts"];
  502.     for (var i = 0; i < scripts.length; ++i)
  503.     {
  504.         if (scripts[i].search(/^file:|chrome:/i) != 0)
  505.         {
  506.             display(getMsg(MSG_ERR_INVALID_SCHEME, scripts[i]), MT_ERROR);
  507.             continue;
  508.         }
  509.  
  510.         var path = getFileFromURLSpec(scripts[i]);
  511.  
  512.         if (!path.exists())
  513.         {
  514.             display(getMsg(MSG_ERR_ITEM_NOT_FOUND, scripts[i]), MT_WARN);
  515.             continue;
  516.         }
  517.  
  518.         if (path.isDirectory())
  519.             loadPluginDirectory(path);
  520.         else
  521.             loadLocalFile(path);
  522.     }
  523. }
  524.  
  525. function loadPluginDirectory(localPath, recurse)
  526. {
  527.     if (typeof recurse == "undefined")
  528.         recurse = 1;
  529.  
  530.     var initPath = localPath.clone();
  531.     initPath.append("init.js");
  532.     if (initPath.exists())
  533.         loadLocalFile(initPath);
  534.  
  535.     if (recurse < 1)
  536.         return;
  537.  
  538.     var enumer = localPath.directoryEntries;
  539.     while (enumer.hasMoreElements())
  540.     {
  541.         var entry = enumer.getNext();
  542.         entry = entry.QueryInterface(Components.interfaces.nsILocalFile);
  543.         if (entry.isDirectory())
  544.             loadPluginDirectory(entry, recurse - 1);
  545.     }
  546. }
  547.  
  548. function loadLocalFile(localFile)
  549. {
  550.     var url = getURLSpecFromFile(localFile);
  551.     var glob = new Object();
  552.     dispatch("load", {url: url, scope: glob});
  553. }
  554.  
  555. function getPluginById(id)
  556. {
  557.     for (var i = 0; i < client.plugins.length; ++i)
  558.     {
  559.         if (client.plugins[i].id == id)
  560.             return client.plugins[i];
  561.  
  562.     }
  563.  
  564.     return null;
  565. }
  566.  
  567. function getPluginIndexById(id)
  568. {
  569.     for (var i = 0; i < client.plugins.length; ++i)
  570.     {
  571.         if (client.plugins[i].id == id)
  572.             return i;
  573.  
  574.     }
  575.  
  576.     return -1;
  577. }
  578.  
  579. function getPluginByURL(url)
  580. {
  581.     for (var i = 0; i < client.plugins.length; ++i)
  582.     {
  583.         if (client.plugins[i].url == url)
  584.             return client.plugins[i];
  585.  
  586.     }
  587.  
  588.     return null;
  589. }
  590.  
  591. function getPluginIndexByURL(url)
  592. {
  593.     for (var i = 0; i < client.plugins.length; ++i)
  594.     {
  595.         if (client.plugins[i].url == url)
  596.             return i;
  597.  
  598.     }
  599.  
  600.     return -1;
  601. }
  602.  
  603. function processStartupURLs()
  604. {
  605.     var wentSomewhere = false;
  606.  
  607.     if ("arguments" in window &&
  608.         0 in window.arguments && typeof window.arguments[0] == "object" &&
  609.         "url" in window.arguments[0])
  610.     {
  611.         var url = window.arguments[0].url;
  612.         if (url.search(/^ircs?:\/?\/?\/?$/i) == -1)
  613.         {
  614.             /* if the url is not irc: irc:/, irc://, or ircs equiv then go to it. */
  615.             gotoIRCURL(url);
  616.             wentSomewhere = true;
  617.         }
  618.     }
  619.  
  620.     if (!wentSomewhere)
  621.     {
  622.         /* if we had nowhere else to go, connect to any default urls */
  623.         var ary = client.prefs["initialURLs"];
  624.         for (var i = 0; i < ary.length; ++i)
  625.         {
  626.             if (ary[i] && ary[i] == "irc:///")
  627.             {
  628.                 // Clean out "default network" entries, which we don't
  629.                 // support any more; replace with the harmless irc:// URL.
  630.                 ary[i] = "irc://";
  631.                 client.prefs["initialURLs"].update();
  632.             }
  633.             if (ary[i] && ary[i] != "irc://")
  634.                 gotoIRCURL(ary[i]);
  635.         }
  636.     }
  637.  
  638.     if (client.viewsArray.length > 1 && !isStartupURL("irc://"))
  639.     {
  640.         dispatch("delete-view", {view: client});
  641.     }
  642. }
  643.  
  644. function destroy()
  645. {
  646.     destroyPrefs();
  647. }
  648.  
  649. function setStatus (str)
  650. {
  651.     client.statusElement.setAttribute ("label", str);
  652.     return str;
  653. }
  654.  
  655. client.__defineSetter__ ("status", setStatus);
  656.  
  657. function getStatus ()
  658. {
  659.     return client.statusElement.getAttribute ("label");
  660. }
  661.  
  662. client.__defineGetter__ ("status", getStatus);
  663.  
  664. function isVisible (id)
  665. {
  666.     var e = document.getElementById(id);
  667.  
  668.     if (!ASSERT(e,"Bogus id ``" + id + "'' passed to isVisible() **"))
  669.         return false;
  670.  
  671.     return (e.getAttribute ("collapsed") != "true");
  672. }
  673.  
  674. client.getConnectedNetworks =
  675. function getConnectedNetworks()
  676. {
  677.     var rv = [];
  678.     for (var n in client.networks)
  679.     {
  680.         if (client.networks[n].isConnected())
  681.             rv.push(client.networks[n]);
  682.     }
  683.     return rv;
  684. }
  685.  
  686. function insertLink (matchText, containerTag)
  687. {
  688.     var href;
  689.     var linkText;
  690.  
  691.     var trailing;
  692.     ary = matchText.match(/([.,?]+)$/);
  693.     if (ary)
  694.     {
  695.         linkText = RegExp.leftContext;
  696.         trailing = ary[1];
  697.     }
  698.     else
  699.     {
  700.         linkText = matchText;
  701.     }
  702.  
  703.     var ary = linkText.match (/^(\w[\w-]+):/);
  704.     if (ary)
  705.     {
  706.         if (!("schemes" in client))
  707.         {
  708.             var pfx = "@mozilla.org/network/protocol;1?name=";
  709.             var len = pfx.length;
  710.  
  711.             client.schemes = new Object();
  712.             for (var c in Components.classes)
  713.             {
  714.                 if (c.indexOf(pfx) == 0)
  715.                     client.schemes[c.substr(len)] = true;
  716.             }
  717.         }
  718.  
  719.         if (!(ary[1] in client.schemes))
  720.         {
  721.             insertHyphenatedWord(matchText, containerTag);
  722.             return;
  723.         }
  724.  
  725.         href = linkText;
  726.     }
  727.     else
  728.     {
  729.         href = "http://" + linkText;
  730.     }
  731.  
  732.     var max = client.prefs["urls.store.max"];
  733.     if (client.prefs["urls.list"].unshift(href) > max)
  734.         client.prefs["urls.list"].pop();
  735.     client.prefs["urls.list"].update();
  736.  
  737.     var anchor = document.createElementNS ("http://www.w3.org/1999/xhtml",
  738.                                            "html:a");
  739.     anchor.setAttribute ("href", href);
  740.     anchor.setAttribute ("class", "chatzilla-link");
  741.     anchor.setAttribute ("target", "_content");
  742.     insertHyphenatedWord (linkText, anchor);
  743.     containerTag.appendChild (anchor);
  744.     if (trailing)
  745.         insertHyphenatedWord (trailing, containerTag);
  746.  
  747. }
  748.  
  749. function insertMailToLink (matchText, containerTag)
  750. {
  751.  
  752.     var href;
  753.  
  754.     if (matchText.indexOf ("mailto:") != 0)
  755.         href = "mailto:" + matchText;
  756.     else
  757.         href = matchText;
  758.  
  759.     var anchor = document.createElementNS ("http://www.w3.org/1999/xhtml",
  760.                                            "html:a");
  761.     anchor.setAttribute ("href", href);
  762.     anchor.setAttribute ("class", "chatzilla-link");
  763.     //anchor.setAttribute ("target", "_content");
  764.     insertHyphenatedWord (matchText, anchor);
  765.     containerTag.appendChild (anchor);
  766.  
  767. }
  768.  
  769. function insertChannelLink (matchText, containerTag, eventData)
  770. {
  771.     var bogusChannels =
  772.         /^#(include|error|define|if|ifdef|else|elsif|endif|\d+)$/i;
  773.  
  774.     if (!("network" in eventData) || !eventData.network ||
  775.         matchText.search(bogusChannels) != -1)
  776.     {
  777.         containerTag.appendChild(document.createTextNode(matchText));
  778.         return;
  779.     }
  780.  
  781.     var encodedMatchText = fromUnicode(matchText, eventData.sourceObject);
  782.     var anchor = document.createElementNS("http://www.w3.org/1999/xhtml",
  783.                                           "html:a");
  784.     anchor.setAttribute ("href", eventData.network.getURL() +
  785.                          ecmaEscape(encodedMatchText));
  786.     anchor.setAttribute ("class", "chatzilla-link");
  787.     insertHyphenatedWord (matchText, anchor);
  788.     containerTag.appendChild (anchor);
  789. }
  790.  
  791. function insertBugzillaLink (matchText, containerTag, eventData)
  792. {
  793.     var number = matchText.match (/(\d+)/)[1];
  794.  
  795.     var anchor = document.createElementNS ("http://www.w3.org/1999/xhtml",
  796.                                            "html:a");
  797.  
  798.     var bugURL;
  799.     if (eventData.channel)
  800.         bugURL = eventData.channel.prefs["bugURL"];
  801.     else if (eventData.network)
  802.         bugURL = eventData.network.prefs["bugURL"];
  803.     else
  804.         bugURL = client.prefs["bugURL"];
  805.  
  806.     anchor.setAttribute ("href", bugURL.replace("%s", number));
  807.     anchor.setAttribute ("class", "chatzilla-link");
  808.     anchor.setAttribute ("target", "_content");
  809.     insertHyphenatedWord (matchText, anchor);
  810.     containerTag.appendChild (anchor);
  811.  
  812. }
  813.  
  814. function insertRheet (matchText, containerTag)
  815. {
  816.  
  817.     var anchor = document.createElementNS ("http://www.w3.org/1999/xhtml",
  818.                                            "html:a");
  819.     anchor.setAttribute ("href",
  820.                          "http://ftp.mozilla.org/pub/mozilla.org/mozilla/libraries/bonus-tracks/rheet.wav");
  821.     anchor.setAttribute ("class", "chatzilla-rheet chatzilla-link");
  822.     //anchor.setAttribute ("target", "_content");
  823.     insertHyphenatedWord (matchText, anchor);
  824.     containerTag.appendChild (anchor);
  825. }
  826.  
  827. function insertQuote (matchText, containerTag)
  828. {
  829.     if (matchText == "``")
  830.         containerTag.appendChild(document.createTextNode("\u201c"));
  831.     else
  832.         containerTag.appendChild(document.createTextNode("\u201d"));
  833. }
  834.  
  835. function insertSmiley(emoticon, containerTag)
  836. {
  837.     var type = "error";
  838.  
  839.     if (emoticon.search(/\>[-^v]?\)/) != -1)
  840.         type = "face-alien";
  841.     else if (emoticon.search(/\>[=:;][-^v]?[(|]/) != -1)
  842.         type = "face-angry";
  843.     else if (emoticon.search(/[=:;][-^v]?[Ss\\\/]/) != -1)
  844.         type = "face-confused";
  845.     else if (emoticon.search(/[B8][-^v]?[)\]]/) != -1)
  846.         type = "face-cool";
  847.     else if (emoticon.search(/[=:;][~'][-^v]?\(/) != -1)
  848.         type = "face-cry";
  849.     else if (emoticon.search(/o[._]O/) != -1)
  850.         type = "face-dizzy";
  851.     else if (emoticon.search(/O[._]o/) != -1)
  852.         type = "face-dizzy-back";
  853.     else if (emoticon.search(/o[._]o|O[._]O/) != -1)
  854.         type = "face-eek";
  855.     else if (emoticon.search(/\>[=:;][-^v]?D/) != -1)
  856.         type = "face-evil";
  857.     else if (emoticon.search(/[=:;][-^v]?DD/) != -1)
  858.         type = "face-lol";
  859.     else if (emoticon.search(/[=:;][-^v]?D/) != -1)
  860.         type = "face-laugh";
  861.     else if (emoticon.search(/\([-^v]?D|[xX][-^v]?D/) != -1)
  862.         type = "face-rofl";
  863.     else if (emoticon.search(/[=:;][-^v]?\|/) != -1)
  864.         type = "face-normal";
  865.     else if (emoticon.search(/[=:;][-^v]?\?/) != -1)
  866.         type = "face-question";
  867.     else if (emoticon.search(/[=:;]"[)\]]/) != -1)
  868.         type = "face-red";
  869.     else if (emoticon.search(/9[._]9/) != -1)
  870.         type = "face-rolleyes";
  871.     else if (emoticon.search(/[=:;][-^v]?[(\[]/) != -1)
  872.         type = "face-sad";
  873.     else if (emoticon.search(/[=:][-^v]?[)\]]/) != -1)
  874.         type = "face-smile";
  875.     else if (emoticon.search(/[=:;][-^v]?[0oO]/) != -1)
  876.         type = "face-surprised";
  877.     else if (emoticon.search(/[=:;][-^v]?[pP]/) != -1)
  878.         type = "face-tongue";
  879.     else if (emoticon.search(/;[-^v]?[)\]]/) != -1)
  880.         type = "face-wink";
  881.  
  882.     if (type == "error")
  883.     {
  884.         // We didn't actually match anything, so it'll be a too-generic match
  885.         // from the munger RegExp.
  886.         containerTag.appendChild(document.createTextNode(emoticon));
  887.         return;
  888.     }
  889.  
  890.     var span = document.createElementNS ("http://www.w3.org/1999/xhtml",
  891.                                          "html:span");
  892.  
  893.     /* create a span to hold the emoticon text */
  894.     span.setAttribute ("class", "chatzilla-emote-txt");
  895.     span.setAttribute ("type", type);
  896.     span.appendChild (document.createTextNode (emoticon));
  897.     containerTag.appendChild (span);
  898.  
  899.     /* create an empty span after the text.  this span will have an image added
  900.      * after it with a chatzilla-emote:after css rule. using
  901.      * chatzilla-emote-txt:after is not good enough because it does not allow us
  902.      * to turn off the emoticon text, but keep the image.  ie.
  903.      * chatzilla-emote-txt { display: none; } turns off
  904.      * chatzilla-emote-txt:after as well.*/
  905.     span = document.createElementNS ("http://www.w3.org/1999/xhtml",
  906.                                      "html:span");
  907.     span.setAttribute ("class", "chatzilla-emote");
  908.     span.setAttribute ("type", type);
  909.     span.setAttribute ("title", emoticon);
  910.     containerTag.appendChild (span);
  911.  
  912. }
  913.  
  914. function mircChangeColor (colorInfo, containerTag, data)
  915. {
  916.     if (!client.enableColors)
  917.         return;
  918.  
  919.     var ary = colorInfo.match (/.(\d{1,2}|)(,(\d{1,2})|)/);
  920.  
  921.     var fgColor = ary[1];
  922.     if (fgColor > 16)
  923.         fgColor &= 16;
  924.  
  925.     switch (fgColor.length)
  926.     {
  927.         case 0:
  928.             delete data.currFgColor;
  929.             delete data.currBgColor;
  930.             return;
  931.  
  932.         case 1:
  933.             data.currFgColor = "0" + fgColor;
  934.             break;
  935.  
  936.         case 2:
  937.             data.currFgColor = fgColor;
  938.             break;
  939.     }
  940.  
  941.     if (fgColor == 1)
  942.         delete data.currFgColor;
  943.     if (arrayHasElementAt(ary, 3))
  944.     {
  945.         var bgColor = ary[3];
  946.         if (bgColor > 16)
  947.             bgColor &= 16;
  948.  
  949.         if (bgColor.length == 1)
  950.             data.currBgColor = "0" + bgColor;
  951.         else
  952.             data.currBgColor = bgColor;
  953.  
  954.         if (bgColor == 0)
  955.             delete data.currBgColor;
  956.     }
  957.  
  958.     data.hasColorInfo = true;
  959. }
  960.  
  961. function mircToggleBold (colorInfo, containerTag, data)
  962. {
  963.     if (!client.enableColors)
  964.         return;
  965.  
  966.     if ("isBold" in data)
  967.         delete data.isBold;
  968.     else
  969.         data.isBold = true;
  970.     data.hasColorInfo = true;
  971. }
  972.  
  973. function mircToggleUnder (colorInfo, containerTag, data)
  974. {
  975.     if (!client.enableColors)
  976.         return;
  977.  
  978.     if ("isUnderline" in data)
  979.         delete data.isUnderline;
  980.     else
  981.         data.isUnderline = true;
  982.     data.hasColorInfo = true;
  983. }
  984.  
  985. function mircResetColor (text, containerTag, data)
  986. {
  987.     if (!client.enableColors || !("hasColorInfo" in data))
  988.         return;
  989.  
  990.     delete data.currFgColor;
  991.     delete data.currBgColor;
  992.     delete data.isBold;
  993.     delete data.isUnder;
  994.     delete data.hasColorInfo;
  995. }
  996.  
  997. function mircReverseColor (text, containerTag, data)
  998. {
  999.     if (!client.enableColors)
  1000.         return;
  1001.  
  1002.     var tempColor = ("currFgColor" in data ? data.currFgColor : "01");
  1003.  
  1004.     if ("currBgColor" in data)
  1005.         data.currFgColor = data.currBgColor;
  1006.     else
  1007.         data.currFgColor = "00";
  1008.     data.currBgColor = tempColor;
  1009.     data.hasColorInfo = true;
  1010. }
  1011.  
  1012. function showCtrlChar(c, containerTag)
  1013. {
  1014.     var span = document.createElementNS ("http://www.w3.org/1999/xhtml",
  1015.                                          "html:span");
  1016.     span.setAttribute ("class", "chatzilla-control-char");
  1017.     if (c == "\t")
  1018.     {
  1019.         containerTag.appendChild(document.createTextNode(c));
  1020.         return;
  1021.     }
  1022.  
  1023.     var ctrlStr = c.charCodeAt(0).toString(16);
  1024.     if (ctrlStr.length < 2)
  1025.         ctrlStr = "0" + ctrlStr;
  1026.     span.appendChild (document.createTextNode ("0x" + ctrlStr));
  1027.     containerTag.appendChild (span);
  1028. }
  1029.  
  1030. function insertHyphenatedWord (longWord, containerTag)
  1031. {
  1032.     var wordParts = splitLongWord (longWord, client.MAX_WORD_DISPLAY);
  1033.     for (var i = 0; i < wordParts.length; ++i)
  1034.     {
  1035.         containerTag.appendChild (document.createTextNode (wordParts[i]));
  1036.         if (i != wordParts.length)
  1037.         {
  1038.             var wbr = document.createElementNS ("http://www.w3.org/1999/xhtml",
  1039.                                                 "html:wbr");
  1040.             containerTag.appendChild (wbr);
  1041.         }
  1042.     }
  1043. }
  1044.  
  1045. function insertInlineButton(text, containerTag, data)
  1046. {
  1047.     var ary = text.match(/\[\[([^\]]+)\]\[([^\]]+)\]\[([^\]]+)\]\]/);
  1048.  
  1049.     if (!ary)
  1050.     {
  1051.         containerTag.appendChild(document.createTextNode(text));
  1052.         return;
  1053.     }
  1054.  
  1055.     var label = ary[1];
  1056.     var title = ary[2];
  1057.     var command = ary[3];
  1058.  
  1059.     var link = document.createElementNS("http://www.w3.org/1999/xhtml", "a");
  1060.     link.setAttribute("href", "x-cz-command:" + encodeURI(command));
  1061.     link.setAttribute("title", title);
  1062.     link.appendChild(document.createTextNode(label));
  1063.  
  1064.     containerTag.appendChild(document.createTextNode("["));
  1065.     containerTag.appendChild(link);
  1066.     containerTag.appendChild(document.createTextNode("]"));
  1067. }
  1068.  
  1069. function combineNicks(nickList, max)
  1070. {
  1071.     if (!max)
  1072.         max = 4;
  1073.  
  1074.     var combinedList = [];
  1075.  
  1076.     for (var i = 0; i < nickList.length; i += max)
  1077.     {
  1078.         count = Math.min(max, nickList.length - i);
  1079.         var nicks = nickList.slice(i, i + count);
  1080.         var str = new String(nicks.join(" "));
  1081.         str.count = count;
  1082.         combinedList.push(str);
  1083.     }
  1084.  
  1085.     return combinedList;
  1086. }
  1087.  
  1088. function updateAllStalkExpressions()
  1089. {
  1090.     var list = client.prefs["stalkWords"];
  1091.  
  1092.     for (var name in client.networks)
  1093.     {
  1094.         if ("stalkExpression" in client.networks[name])
  1095.             updateStalkExpression(client.networks[name], list);
  1096.     }
  1097. }
  1098.  
  1099. function updateStalkExpression(network)
  1100. {
  1101.     function escapeChar(ch)
  1102.     {
  1103.         return "\\" + ch;
  1104.     };
  1105.  
  1106.     var list = client.prefs["stalkWords"];
  1107.  
  1108.     var ary = new Array();
  1109.  
  1110.     ary.push(network.primServ.me.unicodeName.replace(/[^\w\d]/g, escapeChar));
  1111.  
  1112.     for (var i = 0; i < list.length; ++i)
  1113.         ary.push(list[i].replace(/[^\w\d]/g, escapeChar));
  1114.  
  1115.     var re;
  1116.     if (client.prefs["stalkWholeWords"])
  1117.         re = "(^|[\\W\\s])((" + ary.join(")|(") + "))([\\W\\s]|$)";
  1118.     else
  1119.         re = "(" + ary.join(")|(") + ")";
  1120.  
  1121.     network.stalkExpression = new RegExp(re, "i");
  1122. }
  1123.  
  1124. function getDefaultFontSize()
  1125. {
  1126.     const PREF_CTRID = "@mozilla.org/preferences-service;1";
  1127.     const nsIPrefService = Components.interfaces.nsIPrefService;
  1128.     const nsIPrefBranch = Components.interfaces.nsIPrefBranch;
  1129.     const XHTML_NS = "http://www.w3.org/1999/xhtml";
  1130.  
  1131.     var prefSvc = Components.classes[PREF_CTRID].getService(nsIPrefService);
  1132.     var prefBranch = prefSvc.getBranch(null);
  1133.  
  1134.     // PX size pref: font.size.variable.x-western
  1135.     var pxSize = 16;
  1136.     try
  1137.     {
  1138.         pxSize = prefBranch.getIntPref("font.size.variable.x-western");
  1139.     }
  1140.     catch(ex) { }
  1141.  
  1142.     var dpi = 96;
  1143.     try
  1144.     {
  1145.         // Get the DPI the fun way (make Mozilla do the work).
  1146.         var b = document.createElement("box");
  1147.         b.style.width = "1in";
  1148.         dpi = window.getComputedStyle(b, null).width.match(/^\d+/);
  1149.     }
  1150.     catch(ex)
  1151.     {
  1152.         try
  1153.         {
  1154.             // Get the DPI the fun way (make Mozilla do the work).
  1155.             b = document.createElementNS("box", XHTML_NS);
  1156.             b.style.width = "1in";
  1157.             dpi = window.getComputedStyle(b, null).width.match(/^\d+/);
  1158.         }
  1159.         catch(ex) { }
  1160.     }
  1161.  
  1162.     return Math.round((pxSize / dpi) * 72);
  1163. }
  1164.  
  1165. function getDefaultContext(cx)
  1166. {
  1167.     if (!cx)
  1168.         cx = new Object();
  1169.     /* Use __proto__ here and in all other get*Context so that the command can
  1170.      * tell the difference between getObjectDetails and actual parameters. See
  1171.      * cmdJoin for more details.
  1172.      */
  1173.     cx.__proto__ = getObjectDetails(client.currentObject);
  1174.     return cx;
  1175. }
  1176.  
  1177. function getMessagesContext(cx, element)
  1178. {
  1179.     if (!cx)
  1180.         cx = new Object();
  1181.     cx.__proto__ = getObjectDetails(client.currentObject);
  1182.     if (!element)
  1183.         element = document.popupNode;
  1184.  
  1185.     while (element)
  1186.     {
  1187.         switch (element.localName)
  1188.         {
  1189.             case "a":
  1190.                 var href = element.getAttribute("href");
  1191.                 cx.url = href;
  1192.                 break;
  1193.  
  1194.             case "tr":
  1195.                 var nickname = element.getAttribute("msg-user");
  1196.                 if (!nickname)
  1197.                     break;
  1198.  
  1199.                 // strip out  a potential ME! suffix
  1200.                 var ary = nickname.match(/(\S+)/);
  1201.                 nickname = ary[1];
  1202.  
  1203.                 if (!cx.network)
  1204.                     break;
  1205.  
  1206.                 // NOTE: nickname is the unicodeName here!
  1207.                 if (cx.channel)
  1208.                     cx.user = cx.channel.getUser(nickname);
  1209.                 else
  1210.                     cx.user = cx.network.getUser(nickname);
  1211.  
  1212.                 if (cx.user)
  1213.                 {
  1214.                     cx.nickname = cx.user.unicodeName;
  1215.                     cx.canonNick = cx.user.canonicalName;
  1216.                 }
  1217.                 else
  1218.                 {
  1219.                     cx.nickname = nickname;
  1220.                 }
  1221.                 break;
  1222.         }
  1223.  
  1224.         element = element.parentNode;
  1225.     }
  1226.  
  1227.     return cx;
  1228. }
  1229.  
  1230. function getTabContext(cx, element)
  1231. {
  1232.     if (!cx)
  1233.         cx = new Object();
  1234.     if (!element)
  1235.         element = document.popupNode;
  1236.  
  1237.     while (element)
  1238.     {
  1239.         if (element.localName == "tab")
  1240.             return getObjectDetails(element.view);
  1241.         element = element.parentNode;
  1242.     }
  1243.  
  1244.     return cx;
  1245. }
  1246.  
  1247. function getUserlistContext(cx)
  1248. {
  1249.     if (!cx)
  1250.         cx = new Object();
  1251.     cx.__proto__ = getObjectDetails(client.currentObject);
  1252.     if (!cx.channel)
  1253.         return cx;
  1254.  
  1255.     cx.userList = new Array();
  1256.     cx.nicknameList = new Array();
  1257.     cx.canonNickList = new Array();
  1258.  
  1259.     var tree = document.getElementById("user-list");
  1260.     var rangeCount = tree.view.selection.getRangeCount();
  1261.  
  1262.     for (var i = 0; i < rangeCount; ++i)
  1263.     {
  1264.         var start = {}, end = {};
  1265.         tree.view.selection.getRangeAt(i, start, end);
  1266.  
  1267.         // If they == -1, we've got no selection, so bail.
  1268.         if ((start.value == -1) && (end.value == -1))
  1269.             return cx;
  1270.  
  1271.         for (var k = start.value; k <= end.value; ++k)
  1272.         {
  1273.             var item = tree.contentView.getItemAtIndex(k);
  1274.             var cell = item.firstChild.firstChild;
  1275.             var user = cx.channel.getUser(cell.getAttribute("unicodeName"));
  1276.             if (user)
  1277.             {
  1278.                 cx.userList.push(user);
  1279.                 cx.nicknameList.push(user.unicodeName);
  1280.                 cx.canonNickList.push(user.canonicalName);
  1281.                 if (i == 0 && k == start.value)
  1282.                 {
  1283.                     cx.user = user;
  1284.                     cx.nickname = user.unicodeName;
  1285.                     cx.canonNick = user.canonicalName;
  1286.                 }
  1287.             }
  1288.         }
  1289.     }
  1290.  
  1291.     return cx;
  1292. }
  1293.  
  1294. function getFontContext(cx)
  1295. {
  1296.     if (!cx)
  1297.         cx = new Object();
  1298.     cx.__proto__ = getObjectDetails(client.currentObject);
  1299.     cx.fontSizeDefault = getDefaultFontSize();
  1300.     var view = client;
  1301.  
  1302.     if ("prefs" in cx.sourceObject)
  1303.     {
  1304.         cx.fontFamily = view.prefs["font.family"];
  1305.         if (cx.fontFamily.match(/^(default|(sans-)?serif|monospace)$/))
  1306.             delete cx.fontFamily;
  1307.  
  1308.         cx.fontSize = view.prefs["font.size"];
  1309.         if (cx.fontSize == 0)
  1310.             delete cx.fontSize;
  1311.     }
  1312.  
  1313.     return cx;
  1314. }
  1315.  
  1316. function msgIsImportant (msg, sourceNick, network)
  1317. {
  1318.     /* This is a huge hack, but it works. What we want is to match against the
  1319.      * plain text of a message, ignoring color codes, bold, etc. so we put it
  1320.      * through the munger. This produces a tree of HTML elements, which we use
  1321.      * |.innerHTML| to convert to a textual representation.
  1322.      *
  1323.      * Then we remove all the HTML tags, using a RegExp.
  1324.      *
  1325.      * It certainly isn't ideal, and there has to be a better way, but it:
  1326.      *   a) works, and
  1327.      *   b) is fast enough to not cause problems,
  1328.      * so it will do for now.
  1329.      */
  1330.     var plainMsg = client.munger.munge(msg, null, {});
  1331.     plainMsg = plainMsg.innerHTML.replace(/<[^>]+>/g, "");
  1332.  
  1333.     var re = network.stalkExpression;
  1334.     if (plainMsg.search(re) != -1 || sourceNick && sourceNick.search(re) == 0)
  1335.         return true;
  1336.  
  1337.     return false;
  1338. }
  1339.  
  1340. function isStartupURL(url)
  1341. {
  1342.     return arrayContains(client.prefs["initialURLs"], url);
  1343. }
  1344.  
  1345. function cycleView (amount)
  1346. {
  1347.     var len = client.viewsArray.length;
  1348.     if (len <= 1)
  1349.         return;
  1350.  
  1351.     var tb = getTabForObject (client.currentObject);
  1352.     if (!tb)
  1353.         return;
  1354.  
  1355.     var vk = Number(tb.getAttribute("viewKey"));
  1356.     var destKey = (vk + amount) % len; /* wrap around */
  1357.     if (destKey < 0)
  1358.         destKey += len;
  1359.  
  1360.     dispatch("set-current-view", { view: client.viewsArray[destKey].source });
  1361. }
  1362.  
  1363. // Plays the sound for a particular event on a type of object.
  1364. function playEventSounds(type, event)
  1365. {
  1366.     if (!client.sound || !client.prefs["sound.enabled"])
  1367.         return;
  1368.  
  1369.     // Converts .TYPE values into the event object names.
  1370.     // IRCChannel => channel, IRCUser => user, etc.
  1371.     if (type.match(/^IRC/))
  1372.         type = type.substr(3, type.length).toLowerCase();
  1373.  
  1374.     var ev = type + "." + event;
  1375.  
  1376.     if (ev in client.soundList)
  1377.         return;
  1378.  
  1379.     if (!(("sound." + ev) in client.prefs))
  1380.         return;
  1381.  
  1382.     var s = client.prefs["sound." + ev];
  1383.  
  1384.     if (!s)
  1385.         return;
  1386.  
  1387.     if (client.prefs["sound.overlapDelay"] > 0)
  1388.     {
  1389.         client.soundList[ev] = true;
  1390.         setTimeout("delete client.soundList['" + ev + "']",
  1391.                    client.prefs["sound.overlapDelay"]);
  1392.     }
  1393.  
  1394.     if (event == "start")
  1395.     {
  1396.         blockEventSounds(type, "event");
  1397.         blockEventSounds(type, "chat");
  1398.         blockEventSounds(type, "stalk");
  1399.     }
  1400.  
  1401.     playSounds(s);
  1402. }
  1403.  
  1404. // Blocks a particular type of event sound occuring.
  1405. function blockEventSounds(type, event)
  1406. {
  1407.     if (!client.sound || !client.prefs["sound.enabled"])
  1408.         return;
  1409.  
  1410.     // Converts .TYPE values into the event object names.
  1411.     // IRCChannel => channel, IRCUser => user, etc.
  1412.     if (type.match(/^IRC/))
  1413.         type = type.substr(3, type.length).toLowerCase();
  1414.  
  1415.     var ev = type + "." + event;
  1416.  
  1417.     if (client.prefs["sound.overlapDelay"] > 0)
  1418.     {
  1419.         client.soundList[ev] = true;
  1420.         setTimeout("delete client.soundList['" + ev + "']",
  1421.                    client.prefs["sound.overlapDelay"]);
  1422.     }
  1423. }
  1424.  
  1425. function playSounds(list)
  1426. {
  1427.     var ary = list.split (" ");
  1428.     if (ary.length == 0)
  1429.         return;
  1430.  
  1431.     playSound(ary[0]);
  1432.     for (var i = 1; i < ary.length; ++i)
  1433.         setTimeout(playSound, 250 * i, ary[i]);
  1434. }
  1435.  
  1436. function playSound(file)
  1437. {
  1438.     if (!client.sound || !client.prefs["sound.enabled"] || !file)
  1439.         return;
  1440.  
  1441.     if (file == "beep")
  1442.     {
  1443.         client.sound.beep();
  1444.     }
  1445.     else
  1446.     {
  1447.         try
  1448.         {
  1449.             var uri = client.iosvc.newURI(file, null, null);
  1450.             client.sound.play(uri);
  1451.         }
  1452.         catch (ex)
  1453.         {
  1454.             // ignore exceptions from this pile of code.
  1455.         }
  1456.     }
  1457. }
  1458.  
  1459. /* timer-based mainloop */
  1460. function mainStep()
  1461. {
  1462.     client.eventPump.stepEvents();
  1463.     setTimeout ("mainStep()", client.STEP_TIMEOUT);
  1464. }
  1465.  
  1466. function openQueryTab(server, nick)
  1467. {
  1468.     var user = server.addUser(nick);
  1469.     if (client.globalHistory)
  1470.         client.globalHistory.addPage(user.getURL());
  1471.     if (!("messages" in user))
  1472.     {
  1473.         var value = "";
  1474.         var same = true;
  1475.         for (var c in server.channels)
  1476.         {
  1477.             var chan = server.channels[c];
  1478.             if (!(user.canonicalName in chan.users))
  1479.                 continue;
  1480.             /* This takes a boolean value for each channel (true - channel has
  1481.              * same value as first), and &&-s them all together. Thus, |same|
  1482.              * will tell us, at the end, if all the channels found have the
  1483.              * same value for charset.
  1484.              */
  1485.             if (value)
  1486.                 same = same && (value == chan.prefs["charset"]);
  1487.             else
  1488.                 value = chan.prefs["charset"];
  1489.         }
  1490.         /* If we've got a value, and it's the same accross all channels,
  1491.          * we use it as the *default* for the charset pref. If not, it'll
  1492.          * just keep the "defer" default which pulls it off the network.
  1493.          */
  1494.         if (value && same)
  1495.         {
  1496.             user.prefManager.prefRecords["charset"].defaultValue = value;
  1497.         }
  1498.  
  1499.         user.displayHere (getMsg(MSG_QUERY_OPENED, user.unicodeName));
  1500.     }
  1501.     user.whois();
  1502.     return user;
  1503. }
  1504.  
  1505. function arraySpeak (ary, single, plural)
  1506. {
  1507.     var rv = "";
  1508.     var and = MSG_AND;
  1509.  
  1510.     switch (ary.length)
  1511.     {
  1512.         case 0:
  1513.             break;
  1514.  
  1515.         case 1:
  1516.             rv = ary[0];
  1517.             if (single)
  1518.                 rv += " " + single;
  1519.             break;
  1520.  
  1521.         case 2:
  1522.             rv = ary[0] + " " + and + " " + ary[1];
  1523.             if (plural)
  1524.                 rv += " " + plural;
  1525.             break;
  1526.  
  1527.         default:
  1528.             for (var i = 0; i < ary.length - 1; ++i)
  1529.                 rv += ary[i] + ", ";
  1530.             rv += and + " " + ary[ary.length - 1];
  1531.             if (plural)
  1532.                 rv += " " + plural;
  1533.             break;
  1534.     }
  1535.  
  1536.     return rv;
  1537.  
  1538. }
  1539.  
  1540. function getObjectDetails (obj, rv)
  1541. {
  1542.     if (!rv)
  1543.         rv = new Object();
  1544.  
  1545.     if (!ASSERT(obj && typeof obj == "object",
  1546.                 "INVALID OBJECT passed to getObjectDetails (" + obj + "). **"))
  1547.     {
  1548.         return rv;
  1549.     }
  1550.  
  1551.     rv.sourceObject = obj;
  1552.     rv.TYPE = obj.TYPE;
  1553.     rv.parent = ("parent" in obj) ? obj.parent : null;
  1554.     rv.user = null;
  1555.     rv.channel = null;
  1556.     rv.server = null;
  1557.     rv.network = null;
  1558.  
  1559.     switch (obj.TYPE)
  1560.     {
  1561.         case "IRCChannel":
  1562.             rv.viewType = MSG_CHANNEL;
  1563.             rv.channel = obj;
  1564.             rv.channelName = obj.unicodeName;
  1565.             rv.server = rv.channel.parent;
  1566.             rv.network = rv.server.parent;
  1567.             break;
  1568.  
  1569.         case "IRCUser":
  1570.             rv.viewType = MSG_USER;
  1571.             rv.user = obj;
  1572.             rv.userName = obj.unicodeName;
  1573.             rv.server = rv.user.parent;
  1574.             rv.network = rv.server.parent;
  1575.             break;
  1576.  
  1577.         case "IRCChanUser":
  1578.             rv.viewType = MSG_USER;
  1579.             rv.user = obj;
  1580.             rv.userName = obj.unicodeName;
  1581.             rv.channel = rv.user.parent;
  1582.             rv.server = rv.channel.parent;
  1583.             rv.network = rv.server.parent;
  1584.             break;
  1585.  
  1586.         case "IRCNetwork":
  1587.             rv.network = obj;
  1588.             rv.viewType = MSG_NETWORK;
  1589.             if ("primServ" in rv.network)
  1590.                 rv.server = rv.network.primServ;
  1591.             else
  1592.                 rv.server = null;
  1593.             break;
  1594.  
  1595.         case "IRCClient":
  1596.             rv.viewType = MSG_TAB;
  1597.             break;
  1598.  
  1599.         default:
  1600.             /* no setup for unknown object */
  1601.             break;
  1602.     }
  1603.  
  1604.     if (rv.network)
  1605.         rv.networkName = rv.network.unicodeName;
  1606.  
  1607.     return rv;
  1608.  
  1609. }
  1610.  
  1611. function findDynamicRule (selector)
  1612. {
  1613.     var rules = frames[0].document.styleSheets[1].cssRules;
  1614.  
  1615.     if (selector instanceof RegExp)
  1616.         fun = "search";
  1617.     else
  1618.         fun = "indexOf";
  1619.  
  1620.     for (var i = 0; i < rules.length; ++i)
  1621.     {
  1622.         var rule = rules.item(i);
  1623.         if (rule.selectorText && rule.selectorText[fun](selector) == 0)
  1624.             return {sheet: frames[0].document.styleSheets[1], rule: rule,
  1625.                     index: i};
  1626.     }
  1627.  
  1628.     return null;
  1629. }
  1630.  
  1631. function addDynamicRule (rule)
  1632. {
  1633.     var rules = frames[0].document.styleSheets[1];
  1634.  
  1635.     var pos = rules.cssRules.length;
  1636.     rules.insertRule (rule, pos);
  1637. }
  1638.  
  1639. function getCommandEnabled(command)
  1640. {
  1641.     try {
  1642.         var dispatcher = top.document.commandDispatcher;
  1643.         var controller = dispatcher.getControllerForCommand(command);
  1644.  
  1645.         return controller.isCommandEnabled(command);
  1646.     }
  1647.     catch (e)
  1648.     {
  1649.         return false;
  1650.     }
  1651. }
  1652.  
  1653. function doCommand(command)
  1654. {
  1655.     try {
  1656.         var dispatcher = top.document.commandDispatcher;
  1657.         var controller = dispatcher.getControllerForCommand(command);
  1658.         if (controller && controller.isCommandEnabled(command))
  1659.             controller.doCommand(command);
  1660.     }
  1661.     catch (e)
  1662.     {
  1663.     }
  1664. }
  1665.  
  1666. var testURLs =
  1667.     ["irc:", "irc://", "irc:///", "irc:///help", "irc:///help,needkey",
  1668.     "irc://irc.foo.org", "irc://foo:6666",
  1669.     "irc://foo", "irc://irc.foo.org/", "irc://foo:6666/", "irc://foo/",
  1670.     "irc://irc.foo.org/,needpass", "irc://foo/,isserver",
  1671.     "irc://moznet/,isserver", "irc://moznet/",
  1672.     "irc://foo/chatzilla", "irc://foo/chatzilla/",
  1673.     "irc://irc.foo.org/?msg=hello%20there",
  1674.     "irc://irc.foo.org/?msg=hello%20there&ignorethis",
  1675.     "irc://irc.foo.org/%23mozilla,needkey?msg=hello%20there&ignorethis",
  1676.     "invalids",
  1677.     "irc://irc.foo.org/,isnick"];
  1678.  
  1679. function doURLTest()
  1680. {
  1681.     for (var u in testURLs)
  1682.     {
  1683.         dd ("testing url \"" + testURLs[u] + "\"");
  1684.         var o = parseIRCURL(testURLs[u]);
  1685.         if (!o)
  1686.             dd ("PARSE FAILED!");
  1687.         else
  1688.             dd (dumpObjectTree(o));
  1689.         dd ("---");
  1690.     }
  1691. }
  1692.  
  1693. function parseIRCURL (url)
  1694. {
  1695.     var specifiedHost = "";
  1696.  
  1697.     var rv = new Object();
  1698.     rv.spec = url;
  1699.     rv.scheme = url.split(":")[0];
  1700.     rv.host = null;
  1701.     rv.target = "";
  1702.     rv.port = (rv.scheme == "ircs" ? 9999 : 6667);
  1703.     rv.msg = "";
  1704.     rv.pass = null;
  1705.     rv.key = null;
  1706.     rv.charset = null;
  1707.     rv.needpass = false;
  1708.     rv.needkey = false;
  1709.     rv.isnick = false;
  1710.     rv.isserver = false;
  1711.  
  1712.     if (url.search(/^(ircs?:\/?\/?)$/i) != -1)
  1713.         return rv;
  1714.  
  1715.     /* split url into <host>/<everything-else> pieces */
  1716.     var ary = url.match (/^ircs?:\/\/([^\/\s]+)?(\/.*)?\s*$/i);
  1717.     if (!ary || !ary[1])
  1718.     {
  1719.         dd ("parseIRCURL: initial split failed");
  1720.         return null;
  1721.     }
  1722.     var host = ary[1];
  1723.     var rest = arrayHasElementAt(ary, 2) ? ary[2] : "";
  1724.  
  1725.     /* split <host> into server (or network) / port */
  1726.     ary = host.match (/^([^\s\:]+)?(\:\d+)?$/);
  1727.     if (!ary)
  1728.     {
  1729.         dd ("parseIRCURL: host/port split failed");
  1730.         return null;
  1731.     }
  1732.  
  1733.     if (arrayHasElementAt(ary, 2))
  1734.     {
  1735.         if (!arrayHasElementAt(ary, 2))
  1736.         {
  1737.             dd ("parseIRCURL: port with no host");
  1738.             return null;
  1739.         }
  1740.         specifiedHost = rv.host = ary[1].toLowerCase();
  1741.         rv.isserver = true;
  1742.         rv.port = parseInt(ary[2].substr(1));
  1743.     }
  1744.     else if (arrayHasElementAt(ary, 1))
  1745.     {
  1746.         specifiedHost = rv.host = ary[1].toLowerCase();
  1747.         if (specifiedHost.indexOf(".") != -1)
  1748.             rv.isserver = true;
  1749.     }
  1750.  
  1751.     if (rest)
  1752.     {
  1753.         ary = rest.match (/^\/([^\,\?\s\/]*)?\/?(,[^\?]*)?(\?.*)?$/);
  1754.         if (!ary)
  1755.         {
  1756.             dd ("parseIRCURL: rest split failed ``" + rest + "''");
  1757.             return null;
  1758.         }
  1759.  
  1760.         rv.target = arrayHasElementAt(ary, 1) ?
  1761.             ecmaUnescape(ary[1]).replace("\n", "\\n") : "";
  1762.         var i = rv.target.indexOf(" ");
  1763.         if (i != -1)
  1764.             rv.target = rv.target.substr(0, i);
  1765.         var params = arrayHasElementAt(ary, 2) ? ary[2].toLowerCase() : "";
  1766.         var query = arrayHasElementAt(ary, 3) ? ary[3] : "";
  1767.  
  1768.         if (params)
  1769.         {
  1770.             rv.isnick =
  1771.                 (params.search (/,\s*isnick\s*,|,\s*isnick\s*$/) != -1);
  1772.             if (rv.isnick && !rv.target)
  1773.             {
  1774.                 dd ("parseIRCURL: isnick w/o target");
  1775.                 /* isnick w/o a target is bogus */
  1776.                 return null;
  1777.             }
  1778.  
  1779.             if (!rv.isserver)
  1780.             {
  1781.                 rv.isserver =
  1782.                     (params.search (/,\s*isserver\s*,|,\s*isserver\s*$/) != -1);
  1783.             }
  1784.  
  1785.             if (rv.isserver && !specifiedHost)
  1786.             {
  1787.                 dd ("parseIRCURL: isserver w/o host");
  1788.                     /* isserver w/o a host is bogus */
  1789.                 return null;
  1790.             }
  1791.  
  1792.             rv.needpass =
  1793.                 (params.search (/,\s*needpass\s*,|,\s*needpass\s*$/) != -1);
  1794.  
  1795.             rv.needkey =
  1796.                 (params.search (/,\s*needkey\s*,|,\s*needkey\s*$/) != -1);
  1797.  
  1798.         }
  1799.  
  1800.         if (query)
  1801.         {
  1802.             ary = query.substr(1).split("&");
  1803.             while (ary.length)
  1804.             {
  1805.                 var arg = ary.pop().split("=");
  1806.                 /*
  1807.                  * we don't want to accept *any* query, or folks could
  1808.                  * say things like "target=foo", and overwrite what we've
  1809.                  * already parsed, so we only use query args we know about.
  1810.                  */
  1811.                 switch (arg[0].toLowerCase())
  1812.                 {
  1813.                     case "msg":
  1814.                         rv.msg = unescape(arg[1]).replace ("\n", "\\n");
  1815.                          break;
  1816.  
  1817.                     case "pass":
  1818.                         rv.needpass = true;
  1819.                         rv.pass = unescape(arg[1]).replace ("\n", "\\n");
  1820.                         break;
  1821.  
  1822.                     case "key":
  1823.                         rv.needkey = true;
  1824.                         rv.key = unescape(arg[1]).replace ("\n", "\\n");
  1825.                         break;
  1826.  
  1827.                     case "charset":
  1828.                         rv.charset = unescape(arg[1]).replace ("\n", "\\n");
  1829.                         break;
  1830.                 }
  1831.             }
  1832.         }
  1833.     }
  1834.  
  1835.     return rv;
  1836.  
  1837. }
  1838.  
  1839. function gotoIRCURL (url)
  1840. {
  1841.     var urlspec = url;
  1842.     if (typeof url == "string")
  1843.         url = parseIRCURL(url);
  1844.  
  1845.     if (!url)
  1846.     {
  1847.         window.alert (getMsg(MSG_ERR_BAD_IRCURL, urlspec));
  1848.         return;
  1849.     }
  1850.  
  1851.     if (!url.host)
  1852.     {
  1853.         /* focus the *client* view for irc:, irc:/, and irc:// (the only irc
  1854.          * urls that don't have a host.  (irc:/// implies a connect to the
  1855.          * default network.)
  1856.          */
  1857.         dispatch("client");
  1858.         return;
  1859.     }
  1860.  
  1861.     var network;
  1862.     var pass = "";
  1863.  
  1864.     if (url.needpass)
  1865.     {
  1866.         if (url.pass)
  1867.             pass = url.pass;
  1868.         else
  1869.             pass = window.promptPassword(getMsg(MSG_URL_PASSWORD, url.spec));
  1870.     }
  1871.  
  1872.     if (url.isserver)
  1873.     {
  1874.         var alreadyThere = false;
  1875.         for (var n in client.networks)
  1876.         {
  1877.             if ((client.networks[n].isConnected()) &&
  1878.                 (client.networks[n].primServ.hostname == url.host) &&
  1879.                 (client.networks[n].primServ.port == url.port))
  1880.             {
  1881.                 /* already connected to this server/port */
  1882.                 network = client.networks[n];
  1883.                 alreadyThere = true;
  1884.                 break;
  1885.             }
  1886.         }
  1887.  
  1888.         if (!alreadyThere)
  1889.         {
  1890.             /*
  1891.             dd ("gotoIRCURL: not already connected to " +
  1892.                 "server " + url.host + " trying to connect...");
  1893.             */
  1894.             network = dispatch((url.scheme == "ircs" ? "sslserver" : "server"),
  1895.                                 {hostname: url.host, port: url.port, password: pass});
  1896.             if (!("pendingURLs" in network))
  1897.                 network.pendingURLs = new Array();
  1898.             network.pendingURLs.unshift(url);
  1899.             return;
  1900.         }
  1901.     }
  1902.     else
  1903.     {
  1904.         /* parsed as a network name */
  1905.         if (!(url.host in client.networks))
  1906.         {
  1907.             display(getMsg(MSG_ERR_UNKNOWN_NETWORK, url.host));
  1908.             return;
  1909.         }
  1910.  
  1911.         network = client.networks[url.host];
  1912.         if (!network.isConnected())
  1913.         {
  1914.             /*
  1915.             dd ("gotoIRCURL: not already connected to " +
  1916.                 "network " + url.host + " trying to connect...");
  1917.             */
  1918.             client.connectToNetwork(network, (url.scheme == "ircs" ? true : false));
  1919.             if (!("pendingURLs" in network))
  1920.                 network.pendingURLs = new Array();
  1921.             network.pendingURLs.unshift(url);
  1922.             return;
  1923.         }
  1924.     }
  1925.  
  1926.     /* already connected, do whatever comes next in the url */
  1927.     //dd ("gotoIRCURL: connected, time to finish parsing ``" + url + "''");
  1928.     if (url.target)
  1929.     {
  1930.         var targetObject;
  1931.         var ev;
  1932.         if (url.isnick)
  1933.         {
  1934.             /* url points to a person. */
  1935.             var nick = url.target;
  1936.             var ary = url.target.split("!");
  1937.             if (ary)
  1938.                 nick = ary[0];
  1939.  
  1940.             targetObject = network.dispatch("query", {nickname: nick});
  1941.         }
  1942.         else
  1943.         {
  1944.             /* url points to a channel */
  1945.             var key;
  1946.             if (url.needkey)
  1947.             {
  1948.                 if (url.key)
  1949.                     key = url.key;
  1950.                 else
  1951.                     key = window.promptPassword(getMsg(MSG_URL_KEY, url.spec));
  1952.             }
  1953.  
  1954.             if (url.charset)
  1955.             {
  1956.                 var d = { channelName: url.target, key: key,
  1957.                           charset: url.charset };
  1958.                 targetObject = network.dispatch("join", d);
  1959.             }
  1960.             else
  1961.             {
  1962.                 // Must do this the hard way... we have the server's format
  1963.                 // for the channel name here, and all our commands only work
  1964.                 // with the Unicode forms.
  1965.                 var serv = network.primServ;
  1966.                 var target = url.target;
  1967.  
  1968.                 /* If we don't have a valid prefix, stick a "#" on it.
  1969.                  * NOTE: This is always a "#" so that URLs may be compared
  1970.                  * properly without involving the server (e.g. off-line).
  1971.                  */
  1972.                 if (arrayIndexOf(serv.channelTypes, target[0]) == -1)
  1973.                     target = "#" + target;
  1974.  
  1975.                 var chan = new CIRCChannel(serv, null, target);
  1976.  
  1977.                 d = { channelName: chan.unicodeName, key: key,
  1978.                       charset: url.charset };
  1979.                 targetObject = network.dispatch("join", d);
  1980.             }
  1981.  
  1982.             if (!targetObject)
  1983.                 return;
  1984.         }
  1985.  
  1986.         if (url.msg)
  1987.         {
  1988.             var msg;
  1989.             if (url.msg.indexOf("\01ACTION") == 0)
  1990.             {
  1991.                 msg = filterOutput(url.msg, "ACTION", "ME!");
  1992.                 targetObject.display(msg, "ACTION", "ME!",
  1993.                                      client.currentObject);
  1994.             }
  1995.             else
  1996.             {
  1997.                 msg = filterOutput(url.msg, "PRIVMSG", "ME!");
  1998.                 targetObject.display(msg, "PRIVMSG", "ME!",
  1999.                                      client.currentObject);
  2000.             }
  2001.             targetObject.say(msg);
  2002.             dispatch("set-current-view", { view: targetObject });
  2003.         }
  2004.     }
  2005.     else
  2006.     {
  2007.         if (!network.messages)
  2008.             network.displayHere (getMsg(MSG_NETWORK_OPENED, network.unicodeName));
  2009.         dispatch("set-current-view", { view: network });
  2010.     }
  2011. }
  2012.  
  2013. function setTopicText (text)
  2014. {
  2015.     var topic = client.statusBar["channel-topic"];
  2016.     var span = document.createElementNS ("http://www.w3.org/1999/xhtml",
  2017.                                          "html:span");
  2018.  
  2019.     span.appendChild(stringToMsg(text, client.currentObject));
  2020.     topic.removeChild(topic.firstChild);
  2021.     topic.appendChild(span);
  2022. }
  2023.  
  2024. function updateProgress()
  2025. {
  2026.     var busy;
  2027.     var progress = -1;
  2028.  
  2029.     if ("busy" in client.currentObject)
  2030.         busy = client.currentObject.busy;
  2031.  
  2032.     if ("progress" in client.currentObject)
  2033.         progress = client.currentObject.progress;
  2034.  
  2035.     client.progressPanel.collapsed = !busy;
  2036.     if (busy && (progress >= 0))
  2037.     {
  2038.         client.progressBar.value = progress;
  2039.         client.progressBar.mode = "determined";
  2040.     }
  2041.     else
  2042.     {
  2043.         client.progressBar.mode = "undetermined";
  2044.     }
  2045. }
  2046.  
  2047. function updateSecurityIcon()
  2048. {
  2049.     var o = getObjectDetails(client.currentObject);
  2050.     var securityButton = window.document.getElementById("security-button");
  2051.     securityButton.firstChild.value = "";
  2052.     securityButton.removeAttribute("level");
  2053.     securityButton.removeAttribute("tooltiptext");
  2054.     if (!o.server || !o.server.isConnected) // No server or connection?
  2055.     {
  2056.         securityButton.setAttribute("tooltiptext", MSG_SECURITY_INFO);
  2057.         return;
  2058.     }
  2059.  
  2060.     var securityState = o.server.connection.getSecurityState()
  2061.     switch (securityState[0]) 
  2062.     {
  2063.         case STATE_IS_SECURE:
  2064.             securityButton.firstChild.value = o.server.hostname;
  2065.             if (securityState[1] == STATE_SECURE_HIGH)
  2066.                 securityButton.setAttribute("level", "high");
  2067.             else // Because low security is the worst we have when being secure
  2068.                 securityButton.setAttribute("level", "low");
  2069.  
  2070.             // Add the tooltip:
  2071.             var issuer = o.server.connection.getCertificate().issuerOrganization;
  2072.             var tooltiptext = getMsg(MSG_SECURE_CONNECTION, issuer);
  2073.             securityButton.setAttribute("tooltiptext", tooltiptext);
  2074.             securityButton.firstChild.setAttribute("tooltiptext", tooltiptext);
  2075.             securityButton.lastChild.setAttribute("tooltiptext", tooltiptext);
  2076.             break;
  2077.         case STATE_IS_BROKEN:
  2078.             securityButton.setAttribute("level", "broken");
  2079.             // No break to make sure we get the correct tooltip
  2080.         case STATE_IS_INSECURE:
  2081.         default:
  2082.             securityButton.setAttribute("tooltiptext", MSG_SECURITY_INFO);
  2083.     }
  2084. }
  2085.  
  2086. function updateNetwork()
  2087. {
  2088.     var o = getObjectDetails (client.currentObject);
  2089.  
  2090.     var lag = MSG_UNKNOWN;
  2091.     var nick = "";
  2092.     if (o.server)
  2093.     {
  2094.         if (o.server.me)
  2095.             nick = o.server.me.unicodeName;
  2096.         lag = (o.server.lag != -1) ? o.server.lag : MSG_UNKNOWN;
  2097.     }
  2098.     client.statusBar["header-url"].setAttribute("value",
  2099.                                                  client.currentObject.getURL());
  2100.     client.statusBar["header-url"].setAttribute("href",
  2101.                                                  client.currentObject.getURL());
  2102.     client.statusBar["header-url"].setAttribute("name",
  2103.                                                  client.currentObject.unicodeName);
  2104. }
  2105.  
  2106. function updateTitle (obj)
  2107. {
  2108.     if (!(("currentObject" in client) && client.currentObject) ||
  2109.         (obj && obj != client.currentObject))
  2110.         return;
  2111.  
  2112.     var tstring;
  2113.     var o = getObjectDetails(client.currentObject);
  2114.     var net = o.network ? o.network.unicodeName : "";
  2115.     var nick = "";
  2116.  
  2117.     switch (client.currentObject.TYPE)
  2118.     {
  2119.         case "IRCNetwork":
  2120.             var serv = "", port = "";
  2121.             if (client.currentObject.isConnected())
  2122.             {
  2123.                 serv = o.server.hostname;
  2124.                 port = o.server.port;
  2125.                 if (o.server.me)
  2126.                     nick = o.server.me.unicodeName;
  2127.                 tstring = getMsg(MSG_TITLE_NET_ON, [nick, net, serv, port]);
  2128.             }
  2129.             else
  2130.             {
  2131.                 nick = client.currentObject.INITIAL_NICK;
  2132.                 tstring = getMsg(MSG_TITLE_NET_OFF, [nick, net]);
  2133.             }
  2134.             break;
  2135.  
  2136.         case "IRCChannel":
  2137.             var chan = "", mode = "", topic = "";
  2138.             if ("me" in o.parent)
  2139.             {
  2140.                 nick = o.parent.me.unicodeName;
  2141.                 if (o.parent.me.canonicalName in client.currentObject.users)
  2142.                 {
  2143.                     var cuser = client.currentObject.users[o.parent.me.canonicalName];
  2144.                     if (cuser.isOp)
  2145.                         nick = "@" + nick;
  2146.                     else if (cuser.isHalfOp)
  2147.                         nick = "%" + nick;
  2148.                     else if (cuser.isVoice)
  2149.                         nick = "+" + nick;
  2150.                 }
  2151.             }
  2152.             else
  2153.             {
  2154.                 nick = MSG_TITLE_NONICK;
  2155.             }
  2156.             chan = o.channel.unicodeName;
  2157.             mode = o.channel.mode.getModeStr();
  2158.             if (!mode)
  2159.                 mode = MSG_TITLE_NO_MODE;
  2160.             topic = o.channel.topic ? o.channel.topic : MSG_TITLE_NO_TOPIC;
  2161.             var re = /\x1f|\x02|\x0f|\x16|\x03([0-9]{1,2}(,[0-9]{1,2})?)?/g;
  2162.             topic = topic.replace(re, "");
  2163.  
  2164.             tstring = getMsg(MSG_TITLE_CHANNEL, [nick, chan, mode, topic]);
  2165.             break;
  2166.  
  2167.         case "IRCUser":
  2168.             nick = client.currentObject.unicodeName;
  2169.             var source = "";
  2170.             if (client.currentObject.name)
  2171.             {
  2172.                 source = "<" + client.currentObject.name + "@" +
  2173.                     client.currentObject.host +">";
  2174.             }
  2175.             tstring = getMsg(MSG_TITLE_USER, [nick, source]);
  2176.             nick = "me" in o.parent ? o.parent.me.unicodeName : MSG_TITLE_NONICK;
  2177.             break;
  2178.  
  2179.         case "IRCClient":
  2180.             nick = client.prefs["nickname"];
  2181.  
  2182.         default:
  2183.             tstring = MSG_TITLE_UNKNOWN;
  2184.             break;
  2185.     }
  2186.  
  2187.     if (0 && !client.uiState["tabstrip"])
  2188.     {
  2189.         var actl = new Array();
  2190.         for (var i in client.activityList)
  2191.             actl.push ((client.activityList[i] == "!") ?
  2192.                        (Number(i) + 1) + "!" : (Number(i) + 1));
  2193.         if (actl.length > 0)
  2194.             tstring = getMsg(MSG_TITLE_ACTIVITY,
  2195.                              [tstring, actl.join (MSG_COMMASP)]);
  2196.     }
  2197.  
  2198.     document.title = tstring;
  2199.     client.statusBar["server-nick"].setAttribute("label", nick);
  2200. }
  2201.  
  2202. function multilineInputMode (state)
  2203. {
  2204.     var multiInput = document.getElementById("multiline-input");
  2205.     var multiInputBox = document.getElementById("multiline-box");
  2206.     var singleInput = document.getElementById("input");
  2207.     var singleInputBox = document.getElementById("singleline-box");
  2208.     var splitter = document.getElementById("input-splitter");
  2209.     var iw = document.getElementById("input-widgets");
  2210.     var h;
  2211.  
  2212.     client._mlMode = state;
  2213.  
  2214.     if (state)  /* turn on multiline input mode */
  2215.     {
  2216.  
  2217.         h = iw.getAttribute ("lastHeight");
  2218.         if (h)
  2219.             iw.setAttribute ("height", h); /* restore the slider position */
  2220.  
  2221.         singleInputBox.setAttribute ("collapsed", "true");
  2222.         splitter.setAttribute ("collapsed", "false");
  2223.         multiInputBox.setAttribute ("collapsed", "false");
  2224.         // multiInput should have the same direction as singleInput
  2225.         multiInput.setAttribute("dir", singleInput.getAttribute("dir"));
  2226.         client.input = multiInput;
  2227.     }
  2228.     else  /* turn off multiline input mode */
  2229.     {
  2230.         h = iw.getAttribute ("height");
  2231.         iw.setAttribute ("lastHeight", h); /* save the slider position */
  2232.         iw.removeAttribute ("height");     /* let the slider drop */
  2233.  
  2234.         splitter.setAttribute ("collapsed", "true");
  2235.         multiInputBox.setAttribute ("collapsed", "true");
  2236.         singleInputBox.setAttribute ("collapsed", "false");
  2237.         // singleInput should have the same direction as multiInput
  2238.         singleInput.setAttribute("dir", multiInput.getAttribute("dir"));
  2239.         client.input = singleInput;
  2240.     }
  2241.  
  2242.     client.input.focus();
  2243. }
  2244.  
  2245. function displayCertificateInfo()
  2246. {
  2247.     var o = getObjectDetails(client.currentObject);
  2248.     if (!o.server)
  2249.         return;
  2250.  
  2251.     if (!o.server.isSecure)
  2252.     {
  2253.         alert(getMsg(MSG_INSECURE_SERVER, o.server.hostname));
  2254.         return;
  2255.     }
  2256.  
  2257.     viewCert(o.server.connection.getCertificate());
  2258. }
  2259.  
  2260. function newInlineText (data, className, tagName)
  2261. {
  2262.     if (typeof tagName == "undefined")
  2263.         tagName = "html:span";
  2264.  
  2265.     var a = document.createElementNS ("http://www.w3.org/1999/xhtml",
  2266.                                       tagName);
  2267.     if (className)
  2268.         a.setAttribute ("class", className);
  2269.  
  2270.     switch (typeof data)
  2271.     {
  2272.         case "string":
  2273.             a.appendChild (document.createTextNode (data));
  2274.             break;
  2275.  
  2276.         case "object":
  2277.             for (var p in data)
  2278.                 if (p != "data")
  2279.                     a.setAttribute (p, data[p]);
  2280.                 else
  2281.                     a.appendChild (document.createTextNode (data[p]));
  2282.             break;
  2283.  
  2284.         case "undefined":
  2285.             break;
  2286.  
  2287.         default:
  2288.             ASSERT(0, "INVALID TYPE ('" + typeof data + "') passed to " +
  2289.                    "newInlineText.");
  2290.             break;
  2291.  
  2292.     }
  2293.  
  2294.     return a;
  2295.  
  2296. }
  2297.  
  2298. function stringToMsg (message, obj)
  2299. {
  2300.     var ary = message.split ("\n");
  2301.     var span = document.createElementNS ("http://www.w3.org/1999/xhtml",
  2302.                                          "html:span");
  2303.     var data = getObjectDetails(obj);
  2304.  
  2305.     if (ary.length == 1)
  2306.         client.munger.munge(ary[0], span, data);
  2307.     else
  2308.     {
  2309.         for (var l = 0; l < ary.length - 1; ++l)
  2310.         {
  2311.             client.munger.munge(ary[l], span, data);
  2312.             span.appendChild
  2313.                 (document.createElementNS ("http://www.w3.org/1999/xhtml",
  2314.                                            "html:br"));
  2315.         }
  2316.         client.munger.munge(ary[l], span, data);
  2317.     }
  2318.  
  2319.     return span;
  2320. }
  2321.  
  2322. function getFrame()
  2323. {
  2324.     if (client.deck.childNodes.length == 0)
  2325.         return undefined;
  2326.     var panel = client.deck.selectedPanel;
  2327.     return getContentWindow(panel);
  2328. }
  2329.  
  2330. client.__defineGetter__ ("currentFrame", getFrame);
  2331.  
  2332. function setCurrentObject (obj)
  2333. {
  2334.     function clearList()
  2335.     {
  2336.         client.rdf.Unassert (client.rdf.resNullChan, client.rdf.resChanUser,
  2337.                              client.rdf.resNullUser, true);
  2338.     };
  2339.  
  2340.     if (!ASSERT(obj.messages, "INVALID OBJECT passed to setCurrentObject **"))
  2341.         return;
  2342.  
  2343.     if ("currentObject" in client && client.currentObject == obj)
  2344.         return;
  2345.  
  2346.     var tb, userList;
  2347.  
  2348.     if ("currentObject" in client && client.currentObject)
  2349.     {
  2350.         tb = getTabForObject(client.currentObject);
  2351.     }
  2352.     if (tb)
  2353.     {
  2354.         tb.selected = false;
  2355.         tb.setAttribute ("state", "normal");
  2356.     }
  2357.  
  2358.     /* Unselect currently selected users. */
  2359.     userList = document.getElementById("user-list");
  2360.     /* If the splitter's collapsed, the userlist *isn't* visible, but we'll not
  2361.      * get told when it becomes visible, so update it even if it's only the
  2362.      * splitter visible. */
  2363.     if (isVisible("user-list-box") || isVisible("main-splitter"))
  2364.     {
  2365.         /* Remove currently selected items before this tree gets rerooted,
  2366.          * because it seems to remember the selections for eternity if not. */
  2367.         if (userList.view && userList.view.selection)
  2368.             userList.view.selection.select(-1);
  2369.  
  2370.         if (obj.TYPE == "IRCChannel")
  2371.         {
  2372.             client.rdf.setTreeRoot("user-list", obj.getGraphResource());
  2373.             updateUserList();
  2374.         }
  2375.         else
  2376.         {
  2377.             var rdf = client.rdf;
  2378.             rdf.setTreeRoot("user-list", rdf.resNullChan);
  2379.             rdf.Assert (rdf.resNullChan, rdf.resChanUser, rdf.resNullUser,
  2380.                         true);
  2381.             setTimeout(clearList, 100);
  2382.         }
  2383.     }
  2384.  
  2385.     client.currentObject = obj;
  2386.     tb = dispatch("create-tab-for-view", { view: obj });
  2387.     if (tb)
  2388.     {
  2389.         tb.selected = true;
  2390.         tb.setAttribute ("state", "current");
  2391.     }
  2392.  
  2393.     var vk = Number(tb.getAttribute("viewKey"));
  2394.     delete client.activityList[vk];
  2395.     client.deck.selectedIndex = vk;
  2396.  
  2397.     updateTitle();
  2398.     updateProgress();
  2399.     updateSecurityIcon();
  2400.  
  2401.     if (client.PRINT_DIRECTION == 1)
  2402.         scrollDown(obj.frame, false);
  2403.  
  2404.     // Input area should have the same direction as the output area
  2405.     if (("frame" in client.currentObject) &&
  2406.         client.currentObject.frame &&
  2407.         ("contentDocument" in client.currentObject.frame) &&
  2408.         client.currentObject.frame.contentDocument &&
  2409.         ("body" in client.currentObject.frame.contentDocument) &&
  2410.         client.currentObject.frame.contentDocument.body)
  2411.     {
  2412.         var contentArea = client.currentObject.frame.contentDocument.body;
  2413.         client.input.setAttribute("dir", contentArea.getAttribute("dir"));
  2414.     }
  2415.     client.input.focus();
  2416. }
  2417.  
  2418. function checkScroll(frame)
  2419. {
  2420.     var window = getContentWindow(frame);
  2421.     if (!window || !("document" in window))
  2422.         return false;
  2423.  
  2424.     return (window.document.height - window.innerHeight -
  2425.             window.pageYOffset) < 160;
  2426. }
  2427.  
  2428. function scrollDown(frame, force)
  2429. {
  2430.     var window = getContentWindow(frame);
  2431.     if (window && (force || checkScroll(frame)))
  2432.         window.scrollTo(0, window.document.height);
  2433. }
  2434.  
  2435. /* valid values for |what| are "superfluous", "activity", and "attention".
  2436.  * final value for state is dependant on priority of the current state, and the
  2437.  * new state. the priority is: normal < superfluous < activity < attention.
  2438.  */
  2439. function setTabState(source, what, callback)
  2440. {
  2441.     if (typeof source != "object")
  2442.     {
  2443.         if (!ASSERT(source in client.viewsArray,
  2444.                     "INVALID SOURCE passed to setTabState"))
  2445.             return;
  2446.         source = client.viewsArray[source].source;
  2447.     }
  2448.  
  2449.     callback = callback || false;
  2450.  
  2451.     var tb = source.dispatch("create-tab-for-view", { view: source });
  2452.     var vk = Number(tb.getAttribute("viewKey"));
  2453.  
  2454.     var current = ("currentObject" in client && client.currentObject == source);
  2455.  
  2456.     /* We want to play sounds if they're from a non-current view, or we don't
  2457.      * have focus at all. Also make sure stalk matches always play sounds.
  2458.      * Also make sure we don't play on the 2nd half of the flash (Callback).
  2459.      */
  2460.     if (!callback && (!window.isFocused || !current || (what == "attention")))
  2461.     {
  2462.         if (what == "attention")
  2463.             playEventSounds(source.TYPE, "stalk");
  2464.         else if (what == "activity")
  2465.             playEventSounds(source.TYPE, "chat");
  2466.         else if (what == "superfluous")
  2467.             playEventSounds(source.TYPE, "event");
  2468.     }
  2469.  
  2470.     // Only change the tab's colour if it's not the active view.
  2471.     if (!current)
  2472.     {
  2473.         var state = tb.getAttribute("state");
  2474.         if (state == what)
  2475.         {
  2476.             /* if the tab state has an equal priority to what we are setting
  2477.              * then blink it */
  2478.             if (client.prefs["activityFlashDelay"] > 0)
  2479.             {
  2480.                 tb.setAttribute("state", "normal");
  2481.                 setTimeout(setTabState, client.prefs["activityFlashDelay"],
  2482.                            vk, what, true);
  2483.             }
  2484.         }
  2485.         else
  2486.         {
  2487.             if (state == "normal" || state == "superfluous" ||
  2488.                (state == "activity" && what == "attention"))
  2489.             {
  2490.                 /* if the tab state has a lower priority than what we are
  2491.                  * setting, change it to the new state */
  2492.                 tb.setAttribute("state", what);
  2493.                 /* we only change the activity list if priority has increased */
  2494.                 if (what == "attention")
  2495.                    client.activityList[vk] = "!";
  2496.                 else if (what == "activity")
  2497.                     client.activityList[vk] = "+";
  2498.                 else
  2499.                 {
  2500.                    /* this is functionally equivalent to "+" for now */
  2501.                    client.activityList[vk] = "-";
  2502.                 }
  2503.                 updateTitle();
  2504.             }
  2505.             else
  2506.             {
  2507.                 /* the current state of the tab has a higher priority than the
  2508.                  * new state.
  2509.                  * blink the new lower state quickly, then back to the old */
  2510.                 if (client.prefs["activityFlashDelay"] > 0)
  2511.                 {
  2512.                     tb.setAttribute("state", what);
  2513.                     setTimeout(setTabState,
  2514.                                client.prefs["activityFlashDelay"], vk,
  2515.                                state, true);
  2516.                 }
  2517.             }
  2518.         }
  2519.     }
  2520. }
  2521.  
  2522. function notifyAttention (source)
  2523. {
  2524.     if (typeof source != "object")
  2525.         source = client.viewsArray[source].source;
  2526.  
  2527.     if (client.currentObject != source)
  2528.     {
  2529.         var tb = dispatch("create-tab-for-view", { view: source });
  2530.         var vk = Number(tb.getAttribute("viewKey"));
  2531.  
  2532.         tb.setAttribute ("state", "attention");
  2533.         client.activityList[vk] = "!";
  2534.         updateTitle();
  2535.     }
  2536.  
  2537.     if (client.prefs["notify.aggressive"])
  2538.         window.getAttention();
  2539.  
  2540. }
  2541.  
  2542. function setDebugMode(mode)
  2543. {
  2544.     if (mode.indexOf("t") != -1)
  2545.         client.debugHook.enabled = true;
  2546.     else
  2547.         client.debugHook.enabled = false;
  2548.  
  2549.     if (mode.indexOf("c") != -1)
  2550.         client.dbgContexts = true;
  2551.     else
  2552.         delete client.dbgContexts;
  2553.  
  2554.     if (mode.indexOf("d") != -1)
  2555.         client.dbgDispatch = true;
  2556.     else
  2557.         delete client.dbgDispatch;
  2558. }
  2559.  
  2560. function setListMode(mode)
  2561. {
  2562.     var elem = document.getElementById("user-list");
  2563.     if (mode)
  2564.         elem.setAttribute("mode", mode);
  2565.     else
  2566.         elem.removeAttribute("mode");
  2567.     updateUserList();
  2568. }
  2569.  
  2570. function updateUserList()
  2571. {
  2572.     var node;
  2573.     var sortDirection;
  2574.  
  2575.     node = document.getElementById("user-list");
  2576.     if (!node.view)
  2577.         return;
  2578.  
  2579.     const nsIXULSortService = Components.interfaces.nsIXULSortService;
  2580.     const isupports_uri = "@mozilla.org/xul/xul-sort-service;1";
  2581.  
  2582.     var xulSortService =
  2583.         Components.classes[isupports_uri].getService(nsIXULSortService);
  2584.     if (!xulSortService)
  2585.         return;
  2586.  
  2587.     var sortResource;
  2588.  
  2589.     if (client.prefs["sortUsersByMode"])
  2590.         sortResource = RES_PFX + "sortname";
  2591.     else
  2592.         sortResource = RES_PFX + "nick";
  2593.  
  2594.     try
  2595.     {
  2596.         if ("sort" in xulSortService)
  2597.             xulSortService.sort(node, sortResource, "ascending");
  2598.         else
  2599.             xulSortService.Sort(node, sortResource, "ascending");
  2600.     }
  2601.     catch(ex)
  2602.     {
  2603.         dd("Exception calling xulSortService.sort()");
  2604.     }
  2605. }
  2606.  
  2607. function getFrameForDOMWindow(window)
  2608. {
  2609.     var frame;
  2610.     for (var i = 0; i < client.deck.childNodes.length; i++)
  2611.     {
  2612.         frame = client.deck.childNodes[i];
  2613.         if (getContentWindow(frame) == window)
  2614.             return frame;
  2615.     }
  2616.     return undefined;
  2617. }
  2618.  
  2619. function replaceColorCodes(msg)
  2620. {
  2621.     // mIRC codes: underline, bold, Original (reset), colors, reverse colors.
  2622.     msg = msg.replace(/(^|[^%])%U/g, "$1\x1f");
  2623.     msg = msg.replace(/(^|[^%])%B/g, "$1\x02");
  2624.     msg = msg.replace(/(^|[^%])%O/g, "$1\x0f");
  2625.     msg = msg.replace(/(^|[^%])%C/g, "$1\x03");
  2626.     msg = msg.replace(/(^|[^%])%R/g, "$1\x16");
  2627.     // %%[UBOCR] --> %[UBOCR].
  2628.     msg = msg.replace(/%(%[UBOCR])/g, "$1");
  2629.  
  2630.     return msg;
  2631. }
  2632.  
  2633. function decodeColorCodes(msg)
  2634. {
  2635.     // %[UBOCR] --> %%[UBOCR].
  2636.     msg = msg.replace(/(%[UBOCR])/g, "%$1");
  2637.     // Put %-codes back in place of special character codes.
  2638.     msg = msg.replace(/\x1f/g, "%U");
  2639.     msg = msg.replace(/\x02/g, "%B");
  2640.     msg = msg.replace(/\x0f/g, "%O");
  2641.     msg = msg.replace(/\x03/g, "%C");
  2642.     msg = msg.replace(/\x16/g, "%R");
  2643.  
  2644.     return msg;
  2645. }
  2646.  
  2647. client.progressListener = new Object();
  2648.  
  2649. client.progressListener.QueryInterface =
  2650. function qi(iid)
  2651. {
  2652.     return this;
  2653. }
  2654.  
  2655. client.progressListener.onStateChange =
  2656. function client_statechange (webProgress, request, stateFlags, status)
  2657. {
  2658.     const nsIWebProgressListener = Components.interfaces.nsIWebProgressListener;
  2659.     const START = nsIWebProgressListener.STATE_START;
  2660.     const STOP = nsIWebProgressListener.STATE_STOP;
  2661.     const IS_NETWORK = nsIWebProgressListener.STATE_IS_NETWORK;
  2662.     const IS_DOCUMENT = nsIWebProgressListener.STATE_IS_DOCUMENT;
  2663.  
  2664.     var frame;
  2665.  
  2666.     // We only care about the initial start of loading, not the loading of
  2667.     // and page sub-components (IS_DOCUMENT, etc.).
  2668.     if ((stateFlags & START) && (stateFlags & IS_NETWORK) &&
  2669.         (stateFlags & IS_DOCUMENT))
  2670.     {
  2671.         frame = getFrameForDOMWindow(webProgress.DOMWindow);
  2672.         if (!frame)
  2673.         {
  2674.             dd("can't find frame for window (start)");
  2675.             try
  2676.             {
  2677.                 webProgress.removeProgressListener(this);
  2678.             }
  2679.             catch(ex)
  2680.             {
  2681.                 dd("Exception removing progress listener (start): " + ex);
  2682.             }
  2683.         }
  2684.     }
  2685.     // We only want to know when the *network* stops, not the page's
  2686.     // individual components (STATE_IS_REQUEST/STATE_IS_DOCUMENT/somesuch).
  2687.     else if ((stateFlags & STOP) && (stateFlags & IS_NETWORK))
  2688.     {
  2689.         frame = getFrameForDOMWindow(webProgress.DOMWindow);
  2690.         if (!frame)
  2691.         {
  2692.             dd("can't find frame for window (stop)");
  2693.             try
  2694.             {
  2695.                 webProgress.removeProgressListener(this);
  2696.             }
  2697.             catch(ex)
  2698.             {
  2699.                 dd("Exception removing progress listener (stop): " + ex);
  2700.             }
  2701.         }
  2702.         else
  2703.         {
  2704.             var cwin = getContentWindow(frame);
  2705.             if (cwin && "initOutputWindow" in cwin)
  2706.             {
  2707.                 cwin.getMsg = getMsg;
  2708.                 cwin.initOutputWindow(client, frame.source, onMessageViewClick);
  2709.                 cwin.changeCSS(frame.source.getTimestampCSS("data"),
  2710.                                "cz-timestamp-format");
  2711.                 cwin.changeCSS(frame.source.getFontCSS("data"),
  2712.                                "cz-fonts");
  2713.                 scrollDown(frame, true);
  2714.  
  2715.                 try
  2716.                 {
  2717.                     webProgress.removeProgressListener(this);
  2718.                 }
  2719.                 catch(ex)
  2720.                 {
  2721.                     dd("Exception removing progress listener (done): " + ex);
  2722.                 }
  2723.             }
  2724.         }
  2725.     }
  2726. }
  2727.  
  2728. client.progressListener.onProgressChange =
  2729. function client_progresschange (webProgress, request, currentSelf, totalSelf,
  2730.                                 currentMax, selfMax)
  2731. {
  2732. }
  2733.  
  2734. client.progressListener.onLocationChange =
  2735. function client_locationchange (webProgress, request, uri)
  2736. {
  2737. }
  2738.  
  2739. client.progressListener.onStatusChange =
  2740. function client_statuschange (webProgress, request, status, message)
  2741. {
  2742. }
  2743.  
  2744. client.progressListener.onSecurityChange =
  2745. function client_securitychange (webProgress, request, state)
  2746. {
  2747. }
  2748.  
  2749. function syncOutputFrame(obj, nesting)
  2750. {
  2751.     const nsIWebProgress = Components.interfaces.nsIWebProgress;
  2752.     const WINDOW = nsIWebProgress.NOTIFY_STATE_WINDOW;
  2753.     const NETWORK = nsIWebProgress.NOTIFY_STATE_NETWORK;
  2754.     const ALL = nsIWebProgress.NOTIFY_ALL;
  2755.  
  2756.     var iframe = obj.frame;
  2757.  
  2758.     function tryAgain(nLevel)
  2759.     {
  2760.         syncOutputFrame(obj, nLevel);
  2761.     };
  2762.  
  2763.     if (typeof nesting != "number")
  2764.         nesting = 0;
  2765.  
  2766.     if (nesting > 10)
  2767.     {
  2768.         dd("Aborting syncOutputFrame, taken too many tries.");
  2769.         return;
  2770.     }
  2771.  
  2772.     try
  2773.     {
  2774.         if (("contentDocument" in iframe) && ("webProgress" in iframe))
  2775.         {
  2776.             var url = obj.prefs["outputWindowURL"];
  2777.             iframe.addProgressListener(client.progressListener, ALL);
  2778.             iframe.loadURI(url);
  2779.         }
  2780.         else
  2781.         {
  2782.             setTimeout(tryAgain, 500, nesting + 1);
  2783.         }
  2784.     }
  2785.     catch (ex)
  2786.     {
  2787.         dd("caught exception showing session view, will try again later.");
  2788.         dd(dumpObjectTree(ex) + "\n");
  2789.         setTimeout(tryAgain, 500, nesting + 1);
  2790.     }
  2791. }
  2792.  
  2793. function createMessages(source)
  2794. {
  2795.     playEventSounds(source.TYPE, "start");
  2796.  
  2797.     source.messages =
  2798.     document.createElementNS ("http://www.w3.org/1999/xhtml",
  2799.                               "html:table");
  2800.  
  2801.     source.messages.setAttribute ("class", "msg-table");
  2802.     source.messages.setAttribute ("view-type", source.TYPE);
  2803.     var tbody =
  2804.         document.createElementNS ("http://www.w3.org/1999/xhtml",
  2805.                                   "html:tbody");
  2806.     source.messages.appendChild (tbody);
  2807.     source.messageCount = 0;
  2808. }
  2809.  
  2810. /* gets the toolbutton associated with an object
  2811.  * if |create| is present, and true, create if not found */
  2812. function getTabForObject (source, create)
  2813. {
  2814.     var name;
  2815.  
  2816.     if (!ASSERT(source, "UNDEFINED passed to getTabForObject"))
  2817.         return null;
  2818.  
  2819.     if ("viewName" in source)
  2820.     {
  2821.         name = source.viewName;
  2822.     }
  2823.     else
  2824.     {
  2825.         ASSERT(0, "INVALID OBJECT passed to getTabForObject");
  2826.         return null;
  2827.     }
  2828.  
  2829.     var tb, id = "tb[" + name + "]";
  2830.     var matches = 1;
  2831.  
  2832.     for (var i in client.viewsArray)
  2833.     {
  2834.         if (client.viewsArray[i].source == source)
  2835.         {
  2836.             tb = client.viewsArray[i].tb;
  2837.             break;
  2838.         }
  2839.         else
  2840.             if (client.viewsArray[i].tb.getAttribute("id") == id)
  2841.                 id = "tb[" + name + "<" + (++matches) + ">]";
  2842.     }
  2843.  
  2844.     if (!tb && create) /* not found, create one */
  2845.     {
  2846.         if (!("messages" in source) || source.messages == null)
  2847.             createMessages(source);
  2848.         var views = document.getElementById ("views-tbar-inner");
  2849.         tb = document.createElement ("tab");
  2850.         tb.setAttribute("ondraggesture",
  2851.                         "nsDragAndDrop.startDrag(event, tabDNDObserver);");
  2852.         tb.setAttribute("href", source.getURL());
  2853.         tb.setAttribute("name", source.unicodeName);
  2854.         tb.setAttribute("onclick", "onTabClick(event, " + id.quote() + ");");
  2855.         // This wouldn't be here if there was a supported CSS property for it.
  2856.         tb.setAttribute("crop", "center");
  2857.         tb.setAttribute("context", "context:tab");
  2858.         tb.setAttribute("tooltip", "xul-tooltip-node");
  2859.         tb.setAttribute("class", "tab-bottom view-button");
  2860.         tb.setAttribute("id", id);
  2861.         tb.setAttribute("state", "normal");
  2862.  
  2863.         client.viewsArray.push ({source: source, tb: tb});
  2864.         tb.setAttribute ("viewKey", client.viewsArray.length - 1);
  2865.         tb.view = source;
  2866.  
  2867.         if (matches > 1)
  2868.             tb.setAttribute("label", name + "<" + matches + ">");
  2869.         else
  2870.             tb.setAttribute("label", name);
  2871.  
  2872.         views.appendChild(tb);
  2873.  
  2874.         var browser = document.createElement ("browser");
  2875.         browser.setAttribute("class", "output-container");
  2876.         browser.setAttribute("type", "content");
  2877.         browser.setAttribute("flex", "1");
  2878.         browser.setAttribute("tooltip", "html-tooltip-node");
  2879.         browser.setAttribute("context", "context:messages");
  2880.         //browser.setAttribute ("onload", "scrollDown(true);");
  2881.         browser.setAttribute("onclick",
  2882.                              "return onMessageViewClick(event)");
  2883.         browser.setAttribute("ondragover",
  2884.                              "nsDragAndDrop.dragOver(event, " +
  2885.                              "contentDropObserver);");
  2886.         browser.setAttribute("ondragdrop",
  2887.                              "nsDragAndDrop.drop(event, contentDropObserver);");
  2888.         browser.setAttribute("ondraggesture",
  2889.                              "nsDragAndDrop.startDrag(event, " +
  2890.                              "contentAreaDNDObserver);");
  2891.         browser.source = source;
  2892.         source.frame = browser;
  2893.         ASSERT(client.deck, "no deck?");
  2894.         client.deck.appendChild (browser);
  2895.         syncOutputFrame (source);
  2896.     }
  2897.  
  2898.     return tb;
  2899.  
  2900. }
  2901.  
  2902. var contentDropObserver = new Object();
  2903.  
  2904. contentDropObserver.onDragOver =
  2905. function tabdnd_dover (aEvent, aFlavour, aDragSession)
  2906. {
  2907.     if (aEvent.getPreventDefault())
  2908.         return;
  2909.  
  2910.     if (aDragSession.sourceDocument == aEvent.view.document)
  2911.     {
  2912.         aDragSession.canDrop = false;
  2913.         return;
  2914.     }
  2915. }
  2916.  
  2917. contentDropObserver.onDrop =
  2918. function tabdnd_drop (aEvent, aXferData, aDragSession)
  2919. {
  2920.     var url = transferUtils.retrieveURLFromData(aXferData.data,
  2921.                                                 aXferData.flavour.contentType);
  2922.     if (!url || url.search(client.linkRE) == -1)
  2923.         return;
  2924.  
  2925.     if (url.search(/\.css$/i) != -1  && confirm (getMsg(MSG_TABDND_DROP, url)))
  2926.         dispatch("motif", {"motif": url});
  2927.     else if (url.search(/^ircs?:\/\//i) != -1)
  2928.         dispatch("goto-url", {"url": url});
  2929. }
  2930.  
  2931. contentDropObserver.getSupportedFlavours =
  2932. function tabdnd_gsf ()
  2933. {
  2934.     var flavourSet = new FlavourSet();
  2935.     flavourSet.appendFlavour("text/x-moz-url");
  2936.     flavourSet.appendFlavour("application/x-moz-file", "nsIFile");
  2937.     flavourSet.appendFlavour("text/unicode");
  2938.     return flavourSet;
  2939. }
  2940.  
  2941. var tabDNDObserver = new Object();
  2942.  
  2943. tabDNDObserver.onDragStart =
  2944. function tabdnd_dstart (aEvent, aXferData, aDragAction)
  2945. {
  2946.     var tb = aEvent.currentTarget;
  2947.     var href = tb.getAttribute("href");
  2948.     var name = tb.getAttribute("name");
  2949.  
  2950.     aXferData.data = new TransferData();
  2951.     /* x-moz-url has the format "<url>\n<name>", goodie */
  2952.     aXferData.data.addDataForFlavour("text/x-moz-url", href + "\n" + name);
  2953.     aXferData.data.addDataForFlavour("text/unicode", href);
  2954.     aXferData.data.addDataForFlavour("text/html", "<a href='" + href + "'>" +
  2955.                                      name + "</a>");
  2956. }
  2957.  
  2958. function deleteTab (tb)
  2959. {
  2960.     if (!ASSERT(tb.hasAttribute("viewKey"),
  2961.                 "INVALID OBJECT passed to deleteTab (" + tb + ")"))
  2962.     {
  2963.         return null;
  2964.     }
  2965.  
  2966.     var i;
  2967.     var key = Number(tb.getAttribute("viewKey"));
  2968.  
  2969.     /* re-index higher toolbuttons */
  2970.     for (i = key + 1; i < client.viewsArray.length; i++)
  2971.     {
  2972.         client.viewsArray[i].tb.setAttribute ("viewKey", i - 1);
  2973.     }
  2974.     arrayRemoveAt(client.viewsArray, key);
  2975.     var tbinner = document.getElementById("views-tbar-inner");
  2976.     tbinner.removeChild(tb);
  2977.  
  2978.     return key;
  2979. }
  2980.  
  2981. function filterOutput (msg, msgtype)
  2982. {
  2983.     if ("outputFilters" in client)
  2984.     {
  2985.         for (var f in client.outputFilters)
  2986.         {
  2987.             if (client.outputFilters[f].enabled)
  2988.                 msg = client.outputFilters[f].func(msg, msgtype);
  2989.         }
  2990.     }
  2991.  
  2992.     return msg;
  2993. }
  2994.  
  2995. client.addNetwork =
  2996. function cli_addnet(name, serverList, temporary)
  2997. {
  2998.     client.networks[name] =
  2999.         new CIRCNetwork (name, serverList, client.eventPump, temporary);
  3000. }
  3001.  
  3002. client.connectToNetwork =
  3003. function cli_connect(networkOrName, requireSecurity)
  3004. {
  3005.     var network;
  3006.     var name;
  3007.  
  3008.  
  3009.     if (networkOrName instanceof CIRCNetwork)
  3010.     {
  3011.         network = networkOrName;
  3012.     }
  3013.     else
  3014.     {
  3015.         name = networkOrName;
  3016.  
  3017.         if (!(name in client.networks))
  3018.         {
  3019.             display(getMsg(MSG_ERR_UNKNOWN_NETWORK, name), MT_ERROR);
  3020.             return null;
  3021.         }
  3022.  
  3023.         network = client.networks[name];
  3024.     }
  3025.     name = network.unicodeName;
  3026.  
  3027.     if (!("messages" in network))
  3028.         network.displayHere(getMsg(MSG_NETWORK_OPENED, name));
  3029.  
  3030.     dispatch("set-current-view", { view: network });
  3031.  
  3032.     if (network.isConnected())
  3033.     {
  3034.         network.display(getMsg(MSG_ALREADY_CONNECTED, name));
  3035.         return network;
  3036.     }
  3037.  
  3038.     if (network.state != NET_OFFLINE)
  3039.         return network;
  3040.  
  3041.     if (network.prefs["nickname"] == DEFAULT_NICK)
  3042.         network.prefs["nickname"] = prompt(MSG_ENTER_NICK, DEFAULT_NICK);
  3043.  
  3044.     if (!("connecting" in network))
  3045.         network.display(getMsg(MSG_NETWORK_CONNECTING, name));
  3046.  
  3047.     network.connect(requireSecurity);
  3048.  
  3049.     network.updateHeader();
  3050.     client.updateHeader();
  3051.     updateTitle();
  3052.  
  3053.     return network;
  3054. }
  3055.  
  3056.  
  3057. client.getURL =
  3058. function cli_geturl ()
  3059. {
  3060.     return "irc://";
  3061. }
  3062.  
  3063. client.load =
  3064. function cli_load(url, scope)
  3065. {
  3066.     if (!("_loader" in client))
  3067.     {
  3068.         const LOADER_CTRID = "@mozilla.org/moz/jssubscript-loader;1";
  3069.         const mozIJSSubScriptLoader =
  3070.             Components.interfaces.mozIJSSubScriptLoader;
  3071.  
  3072.         var cls;
  3073.         if ((cls = Components.classes[LOADER_CTRID]))
  3074.             client._loader = cls.getService(mozIJSSubScriptLoader);
  3075.     }
  3076.  
  3077.     return client._loader.loadSubScript(url, scope);
  3078. }
  3079.  
  3080. client.sayToCurrentTarget =
  3081. function cli_say(msg)
  3082. {
  3083.     if ("say" in client.currentObject)
  3084.     {
  3085.         msg = filterOutput (msg, "PRIVMSG");
  3086.         display(msg, "PRIVMSG", "ME!", client.currentObject);
  3087.         client.currentObject.say(msg);
  3088.  
  3089.         return;
  3090.     }
  3091.  
  3092.     switch (client.currentObject.TYPE)
  3093.     {
  3094.         case "IRCClient":
  3095.             dispatch("eval", {expression: msg});
  3096.             break;
  3097.  
  3098.         default:
  3099.             if (msg != "")
  3100.             {
  3101.                 display(getMsg(MSG_ERR_NO_DEFAULT, client.currentObject.TYPE),
  3102.                         MT_ERROR);
  3103.             }
  3104.             break;
  3105.     }
  3106. }
  3107.  
  3108. CIRCNetwork.prototype.__defineGetter__("prefs", net_getprefs);
  3109. function net_getprefs()
  3110. {
  3111.     if (!("_prefs" in this))
  3112.     {
  3113.         this._prefManager = getNetworkPrefManager(this);
  3114.         this._prefs = this._prefManager.prefs;
  3115.     }
  3116.  
  3117.     return this._prefs;
  3118. }
  3119.  
  3120. CIRCNetwork.prototype.__defineGetter__("prefManager", net_getprefmgr);
  3121. function net_getprefmgr()
  3122. {
  3123.     if (!("_prefManager" in this))
  3124.     {
  3125.         this._prefManager = getNetworkPrefManager(this);
  3126.         this._prefs = this._prefManager.prefs;
  3127.     }
  3128.  
  3129.     return this._prefManager;
  3130. }
  3131.  
  3132. CIRCServer.prototype.__defineGetter__("prefs", srv_getprefs);
  3133. function srv_getprefs()
  3134. {
  3135.     return this.parent.prefs;
  3136. }
  3137.  
  3138. CIRCServer.prototype.__defineGetter__("prefManager", srv_getprefmgr);
  3139. function srv_getprefmgr()
  3140. {
  3141.     return this.parent.prefManager;
  3142. }
  3143.  
  3144. CIRCChannel.prototype.__defineGetter__("prefs", chan_getprefs);
  3145. function chan_getprefs()
  3146. {
  3147.     if (!("_prefs" in this))
  3148.     {
  3149.         this._prefManager = getChannelPrefManager(this);
  3150.         this._prefs = this._prefManager.prefs;
  3151.     }
  3152.  
  3153.     return this._prefs;
  3154. }
  3155.  
  3156. CIRCChannel.prototype.__defineGetter__("prefManager", chan_getprefmgr);
  3157. function chan_getprefmgr()
  3158. {
  3159.     if (!("_prefManager" in this))
  3160.     {
  3161.         this._prefManager = getChannelPrefManager(this);
  3162.         this._prefs = this._prefManager.prefs;
  3163.     }
  3164.  
  3165.     return this._prefManager;
  3166. }
  3167.  
  3168. CIRCUser.prototype.__defineGetter__("prefs", usr_getprefs);
  3169. function usr_getprefs()
  3170. {
  3171.     if (!("_prefs" in this))
  3172.     {
  3173.         this._prefManager = getUserPrefManager(this);
  3174.         this._prefs = this._prefManager.prefs;
  3175.     }
  3176.  
  3177.     return this._prefs;
  3178. }
  3179.  
  3180. CIRCUser.prototype.__defineGetter__("prefManager", usr_getprefmgr);
  3181. function usr_getprefmgr()
  3182. {
  3183.     if (!("_prefManager" in this))
  3184.     {
  3185.         this._prefManager = getUserPrefManager(this);
  3186.         this._prefs = this._prefManager.prefs;
  3187.     }
  3188.  
  3189.     return this._prefManager;
  3190. }
  3191.  
  3192. CIRCNetwork.prototype.display =
  3193. function net_display (message, msgtype, sourceObj, destObj)
  3194. {
  3195.     var o = getObjectDetails(client.currentObject);
  3196.  
  3197.     if (client.SLOPPY_NETWORKS && client.currentObject != this &&
  3198.         o.network == this && o.server && o.server.isConnected)
  3199.     {
  3200.         client.currentObject.display (message, msgtype, sourceObj, destObj);
  3201.     }
  3202.     else
  3203.     {
  3204.         this.displayHere (message, msgtype, sourceObj, destObj);
  3205.     }
  3206. }
  3207.  
  3208. CIRCUser.prototype.display =
  3209. function usr_display(message, msgtype, sourceObj, destObj)
  3210. {
  3211.     if ("messages" in this)
  3212.     {
  3213.         this.displayHere (message, msgtype, sourceObj, destObj);
  3214.     }
  3215.     else
  3216.     {
  3217.         var o = getObjectDetails(client.currentObject);
  3218.         if (o.server && o.server.isConnected &&
  3219.             o.network == this.parent.parent &&
  3220.             client.currentObject.TYPE != "IRCUser")
  3221.             client.currentObject.display (message, msgtype, sourceObj, destObj);
  3222.         else
  3223.             this.parent.parent.displayHere (message, msgtype, sourceObj,
  3224.                                             destObj);
  3225.     }
  3226. }
  3227.  
  3228. CIRCDCCChat.prototype.display =
  3229. CIRCDCCFileTransfer.prototype.display =
  3230. function dcc_display(message, msgtype, sourceObj, destObj)
  3231. {
  3232.     var o = getObjectDetails(client.currentObject);
  3233.  
  3234.     if ("messages" in this)
  3235.         this.displayHere(message, msgtype, sourceObj, destObj);
  3236.     else
  3237.         client.currentObject.display(message, msgtype, sourceObj, destObj);
  3238. }
  3239.  
  3240. function feedback(e, message, msgtype, sourceObj, destObj)
  3241. {
  3242.     if ("isInteractive" in e && e.isInteractive)
  3243.         display(message, msgtype, sourceObj, destObj);
  3244. }
  3245.  
  3246. CIRCChannel.prototype.feedback =
  3247. CIRCNetwork.prototype.feedback =
  3248. CIRCUser.prototype.feedback =
  3249. CIRCDCCChat.prototype.feedback =
  3250. CIRCDCCFileTransfer.prototype.feedback =
  3251. client.feedback =
  3252. function this_feedback(e, message, msgtype, sourceObj, destObj)
  3253. {
  3254.     if ("isInteractive" in e && e.isInteractive)
  3255.         this.displayHere(message, msgtype, sourceObj, destObj);
  3256. }
  3257.  
  3258. function display (message, msgtype, sourceObj, destObj)
  3259. {
  3260.     client.currentObject.display (message, msgtype, sourceObj, destObj);
  3261. }
  3262.  
  3263. client.getTimestampCSS =
  3264. CIRCNetwork.prototype.getTimestampCSS =
  3265. CIRCChannel.prototype.getTimestampCSS =
  3266. CIRCUser.prototype.getTimestampCSS =
  3267. CIRCDCCChat.prototype.getTimestampCSS =
  3268. CIRCDCCFileTransfer.prototype.getTimestampCSS =
  3269. function this_getTimestampCSS(format)
  3270. {
  3271.     /* Wow, this is cool. We just put together a CSS-rule string based on the
  3272.      * "timestampFormat" preferences. *This* is what CSS is all about. :)
  3273.      * We also provide a "data: URL" format, to simplify other code.
  3274.      */
  3275.     var css;
  3276.  
  3277.     if (this.prefs["timestamps"])
  3278.     {
  3279.         /* Hack. To get around a Mozilla bug, we must force the display back
  3280.          * to a displayed value.
  3281.          */
  3282.         css = ".msg-timestamp { display: table-cell; } " +
  3283.               ".msg-timestamp:before { content: '" +
  3284.               this.prefs["timestampFormat"] + "'; }";
  3285.  
  3286.         var letters = new Array('y', 'm', 'd', 'h', 'n', 's');
  3287.         for (var i = 0; i < letters.length; i++)
  3288.         {
  3289.             css = css.replace("%" + letters[i], "' attr(time-" +
  3290.                               letters[i] + ") '");
  3291.         }
  3292.     }
  3293.     else
  3294.     {
  3295.         /* Completely remove the <td>s if they're off, neatens display. */
  3296.         css = ".msg-timestamp { display: none; }";
  3297.     }
  3298.  
  3299.     if (format == "data")
  3300.         return "data:text/css," + encodeURIComponent(css);
  3301.     return css;
  3302. }
  3303.  
  3304. client.getFontCSS =
  3305. CIRCNetwork.prototype.getFontCSS =
  3306. CIRCChannel.prototype.getFontCSS =
  3307. CIRCUser.prototype.getFontCSS =
  3308. CIRCDCCChat.prototype.getFontCSS =
  3309. CIRCDCCFileTransfer.prototype.getFontCSS =
  3310. function this_getFontCSS(format)
  3311. {
  3312.     /* See this_getTimestampCSS. */
  3313.     var css;
  3314.     var fs;
  3315.     var fn;
  3316.  
  3317.     if (this.prefs["font.family"] != "default")
  3318.         fn = "font-family: " + this.prefs["font.family"] + ";";
  3319.     else
  3320.         fn = "font-family: inherit;";
  3321.     if (this.prefs["font.size"] != 0)
  3322.         fs = "font-size: " + this.prefs["font.size"] + "pt;";
  3323.     else
  3324.         fs = "font-size: medium;";
  3325.  
  3326.     css = "body.chatzilla-body { " + fs + fn + " }";
  3327.  
  3328.     if (format == "data")
  3329.         return "data:text/css," + encodeURIComponent(css);
  3330.     return css;
  3331. }
  3332.  
  3333. client.display =
  3334. client.displayHere =
  3335. CIRCNetwork.prototype.displayHere =
  3336. CIRCChannel.prototype.display =
  3337. CIRCChannel.prototype.displayHere =
  3338. CIRCUser.prototype.displayHere =
  3339. CIRCDCCChat.prototype.displayHere =
  3340. CIRCDCCFileTransfer.prototype.displayHere =
  3341. function __display(message, msgtype, sourceObj, destObj)
  3342. {
  3343.     // We like some control on the number of digits.
  3344.     function formatTimeNumber (num, digits)
  3345.     {
  3346.         var rv = num.toString();
  3347.         while (rv.length < digits)
  3348.             rv = "0" + rv;
  3349.         return rv;
  3350.     };
  3351.  
  3352.     // We need a message type, assume "INFO".
  3353.     if (!msgtype)
  3354.         msgtype = MT_INFO;
  3355.  
  3356.     var blockLevel = false; /* true if this row should be rendered at block
  3357.                              * level, (like, if it has a really long nickname
  3358.                              * that might disturb the rest of the layout)     */
  3359.     var o = getObjectDetails(this);           /* get the skinny on |this|     */
  3360.  
  3361.     // Get the 'me' object, so we can be sure to get the attributes right.
  3362.     var me;
  3363.     if ("me" in this)
  3364.         me = this.me;
  3365.     else if (o.server && "me" in o.server)
  3366.         me = o.server.me;
  3367.  
  3368.     // Let callers get away with "ME!" and we have to substitute here.
  3369.     if (sourceObj == "ME!")
  3370.         sourceObj = me;
  3371.     if (destObj == "ME!")
  3372.         destObj = me;
  3373.  
  3374.     // Get the TYPE of the source object.
  3375.     var fromType = (sourceObj && sourceObj.TYPE) ? sourceObj.TYPE : "unk";
  3376.     // Is the source a user?
  3377.     var fromUser = (fromType.search(/IRC.*User/) != -1);
  3378.     // Get some sort of "name" for the source.
  3379.     var fromAttr = "";
  3380.     if (sourceObj)
  3381.     {
  3382.         if ("unicodeName" in sourceObj)
  3383.             fromAttr = sourceObj.unicodeName;
  3384.         else if ("name" in sourceObj)
  3385.             fromAttr = sourceObj.name;
  3386.         else
  3387.             fromAttr = sourceObj.viewName;
  3388.     }
  3389.     // Attach "ME!" if appropriate, so motifs can style differently.
  3390.     if (sourceObj == me)
  3391.         fromAttr = fromAttr + " ME!";
  3392.  
  3393.     // Get the dest TYPE too...
  3394.     var toType = (destObj) ? destObj.TYPE : "unk";
  3395.     // Is the dest a user?
  3396.     var toUser = (toType.search(/IRC.*User/) != -1);
  3397.     // Get a dest name too...
  3398.     var toAttr = "";
  3399.     if (destObj)
  3400.     {
  3401.         if ("unicodeName" in destObj)
  3402.             toAttr = destObj.unicodeName;
  3403.         else if ("name" in destObj)
  3404.             toAttr = destObj.name;
  3405.         else
  3406.             toAttr = destObj.viewName;
  3407.     }
  3408.     // Also do "ME!" work for the dest.
  3409.     if (destObj && destObj == me)
  3410.         toAttr = me.unicodeName + " ME!";
  3411.  
  3412.     /* isImportant means to style the messages as important, and flash the
  3413.      * window, getAttention means just flash the window. */
  3414.     var isImportant = false, getAttention = false, isSuperfluous = false;
  3415.     var viewType = this.TYPE;
  3416.     var code;
  3417.  
  3418.     var d = new Date();
  3419.     var dateInfo = { y: formatTimeNumber(d.getFullYear(), 4),
  3420.                      m: formatTimeNumber(d.getMonth() + 1, 2),
  3421.                      d: formatTimeNumber(d.getDate(), 2),
  3422.                      h: formatTimeNumber(d.getHours(), 2),
  3423.                      n: formatTimeNumber(d.getMinutes(), 2),
  3424.                      s: formatTimeNumber(d.getSeconds(), 2)
  3425.                    };
  3426.  
  3427.     // Statusbar text, and the line that gets saved to the log.
  3428.     var statusString;
  3429.     var logString;
  3430.  
  3431.     var dtf = client.dtFormatter;
  3432.     var timeStamp = dtf.FormatDateTime("", dtf.dateFormatShort,
  3433.                                        dtf.timeFormatNoSeconds, d.getFullYear(),
  3434.                                        d.getMonth() + 1, d.getDate(),
  3435.                                        d.getHours(), d.getMinutes(),
  3436.                                        d.getSeconds()
  3437.                                       );
  3438.     logString = "[" + timeStamp + "] ";
  3439.  
  3440.     if (fromUser)
  3441.     {
  3442.         statusString = getMsg(MSG_FMT_STATUS,
  3443.                               [timeStamp,
  3444.                                sourceObj.unicodeName + "!" +
  3445.                                sourceObj.name + "@" + sourceObj.host]);
  3446.     }
  3447.     else
  3448.     {
  3449.         var name;
  3450.         if (sourceObj)
  3451.             name = sourceObj.viewName;
  3452.         else
  3453.             name = this.viewName;
  3454.  
  3455.         statusString = getMsg(MSG_FMT_STATUS,
  3456.                               [timeStamp, name]);
  3457.     }
  3458.  
  3459.     // The table row, and it's attributes.
  3460.     var msgRow = document.createElementNS("http://www.w3.org/1999/xhtml",
  3461.                                           "html:tr");
  3462.     msgRow.setAttribute("class", "msg");
  3463.     msgRow.setAttribute("msg-type", msgtype);
  3464.     msgRow.setAttribute("msg-dest", toAttr);
  3465.     msgRow.setAttribute("dest-type", toType);
  3466.     msgRow.setAttribute("view-type", viewType);
  3467.     msgRow.setAttribute("statusText", statusString);
  3468.     if (fromAttr)
  3469.     {
  3470.         if (fromUser)
  3471.             msgRow.setAttribute("msg-user", fromAttr);
  3472.         else
  3473.             msgRow.setAttribute("msg-source", fromAttr);
  3474.     }
  3475.     if (isImportant)
  3476.         msgTimestamp.setAttribute ("important", "true");
  3477.  
  3478.     // Timestamp cell.
  3479.     var msgRowTimestamp = document.createElementNS("http://www.w3.org/1999/xhtml",
  3480.                                                    "html:td");
  3481.     msgRowTimestamp.setAttribute("class", "msg-timestamp");
  3482.     for (var key in dateInfo)
  3483.         msgRowTimestamp.setAttribute("time-" + key, dateInfo[key]);
  3484.  
  3485.     var canMergeData;
  3486.     var msgRowSource, msgRowType, msgRowData;
  3487.     if (fromUser && msgtype.match(/^(PRIVMSG|ACTION|NOTICE)$/))
  3488.     {
  3489.         var nick = sourceObj.unicodeName;
  3490.  
  3491.         var nickURL;
  3492.         if ((sourceObj != me) && ("getURL" in sourceObj))
  3493.             nickURL = sourceObj.getURL();
  3494.  
  3495.         if (sourceObj != me)
  3496.         {
  3497.             // Not from us...
  3498.             if (destObj == me)
  3499.             {
  3500.                 // ...but to us. Messages from someone else to us.
  3501.  
  3502.                 getAttention = true;
  3503.                 this.defaultCompletion = "/msg " + nick + " ";
  3504.  
  3505.                 if (msgtype == "ACTION")
  3506.                 {
  3507.                     logString += "* " + nick + " ";
  3508.                 }
  3509.                 else
  3510.                 {
  3511.                     // If this private message is not in a query view, use
  3512.                     // *nick* instead of <nick>.
  3513.                     if (this.TYPE == "IRCUser")
  3514.                         logString += "<" + nick + "> ";
  3515.                     else
  3516.                         logString += "*" + nick + "* ";
  3517.                 }
  3518.             }
  3519.             else
  3520.             {
  3521.                 // ...or to us. Messages from someone else to channel or similar.
  3522.  
  3523.                 if ((typeof message == "string") && me)
  3524.                 {
  3525.                     isImportant = msgIsImportant(message, nick, o.network);
  3526.                     if (isImportant)
  3527.                     {
  3528.                         this.defaultCompletion = nick +
  3529.                             client.prefs["nickCompleteStr"] + " ";
  3530.                     }
  3531.                 }
  3532.                 if (msgtype == "ACTION")
  3533.                     logString += "* " + nick + " ";
  3534.                 else
  3535.                     logString += "<" + nick + "> ";
  3536.             }
  3537.         }
  3538.         else
  3539.         {
  3540.             // Messages from us to somewhere...
  3541.  
  3542.             if (toUser)
  3543.             {
  3544.                 // From us to a user.
  3545.  
  3546.                 if (this.TYPE == "IRCUser")
  3547.                 {
  3548.                     if (msgtype == "ACTION")
  3549.                         logString += "* " + nick + " ";
  3550.                     else
  3551.                         logString += "<" + nick + "> ";
  3552.                 }
  3553.                 else
  3554.                 {
  3555.                     nick = destObj.unicodeName;
  3556.                     logString += ">" + nick + "< ";
  3557.                 }
  3558.             }
  3559.             else
  3560.             {
  3561.                 if (msgtype == "ACTION")
  3562.                     logString += "*" + nick + " ";
  3563.                 else
  3564.                     logString += "<" + nick + "> ";
  3565.             }
  3566.         }
  3567.  
  3568.         // Mark makes alternate "talkers" show up in different shades.
  3569.         //if (!("mark" in this))
  3570.         //    this.mark = "odd";
  3571.  
  3572.         if (!("lastNickDisplayed" in this) || this.lastNickDisplayed != nick)
  3573.         {
  3574.             this.lastNickDisplayed = nick;
  3575.             this.mark = (("mark" in this) && this.mark == "even") ? "odd" : "even";
  3576.         }
  3577.  
  3578.         msgRowSource = document.createElementNS("http://www.w3.org/1999/xhtml",
  3579.                                              "html:td");
  3580.         msgRowSource.setAttribute("class", "msg-user");
  3581.  
  3582.         // Make excessive nicks get shunted.
  3583.         if (nick && (nick.length > client.MAX_NICK_DISPLAY))
  3584.             blockLevel = true;
  3585.  
  3586.         if (nickURL)
  3587.         {
  3588.             var nick_anchor =
  3589.                 document.createElementNS("http://www.w3.org/1999/xhtml",
  3590.                                          "html:a");
  3591.             nick_anchor.setAttribute("class", "chatzilla-link");
  3592.             nick_anchor.setAttribute("href", nickURL);
  3593.             nick_anchor.appendChild(newInlineText(nick));
  3594.             msgRowSource.appendChild(nick_anchor);
  3595.         }
  3596.         else
  3597.         {
  3598.             msgRowSource.appendChild(newInlineText(nick));
  3599.         }
  3600.         canMergeData = this.prefs["collapseMsgs"];
  3601.     }
  3602.     else
  3603.     {
  3604.         isSuperfluous = true;
  3605.         if (!client.debugHook.enabled && msgtype in client.responseCodeMap)
  3606.         {
  3607.             code = client.responseCodeMap[msgtype];
  3608.         }
  3609.         else
  3610.         {
  3611.             if (!client.debugHook.enabled && client.HIDE_CODES)
  3612.                 code = client.DEFAULT_RESPONSE_CODE;
  3613.             else
  3614.                 code = "[" + msgtype + "]";
  3615.         }
  3616.  
  3617.         /* Display the message code */
  3618.         msgRowType = document.createElementNS("http://www.w3.org/1999/xhtml",
  3619.                                            "html:td");
  3620.         msgRowType.setAttribute("class", "msg-type");
  3621.  
  3622.         msgRowType.appendChild(newInlineText(code));
  3623.         logString += code + " ";
  3624.     }
  3625.  
  3626.     if (message)
  3627.     {
  3628.         msgRowData = document.createElementNS("http://www.w3.org/1999/xhtml",
  3629.                                            "html:td");
  3630.         msgRowData.setAttribute("class", "msg-data");
  3631.  
  3632.         if (typeof message == "string")
  3633.         {
  3634.             msgRowData.appendChild(stringToMsg(message, this));
  3635.             logString += message;
  3636.         }
  3637.         else
  3638.         {
  3639.             msgRowData.appendChild(message);
  3640.             logString += message.innerHTML.replace(/<[^<]*>/g, "");
  3641.         }
  3642.     }
  3643.  
  3644.     if ("mark" in this)
  3645.         msgRow.setAttribute("mark", this.mark);
  3646.  
  3647.     if (isImportant)
  3648.         msgRow.setAttribute ("important", "true");
  3649.  
  3650.     // Timestamps first...
  3651.     msgRow.appendChild(msgRowTimestamp);
  3652.     // Now do the rest of the row, after block-level stuff.
  3653.     if (msgRowSource)
  3654.         msgRow.appendChild(msgRowSource);
  3655.     else
  3656.         msgRow.appendChild(msgRowType);
  3657.     if (msgRowData)
  3658.         msgRow.appendChild(msgRowData);
  3659.  
  3660.     if (blockLevel)
  3661.     {
  3662.         /* putting a div here crashes mozilla, so fake it with nested tables
  3663.          * for now */
  3664.         var tr = document.createElementNS ("http://www.w3.org/1999/xhtml",
  3665.                                            "html:tr");
  3666.         tr.setAttribute ("class", "msg-nested-tr");
  3667.         var td = document.createElementNS ("http://www.w3.org/1999/xhtml",
  3668.                                            "html:td");
  3669.         td.setAttribute ("class", "msg-nested-td");
  3670.         td.setAttribute ("colspan", "3");
  3671.  
  3672.         tr.appendChild(td);
  3673.         var table = document.createElementNS ("http://www.w3.org/1999/xhtml",
  3674.                                               "html:table");
  3675.         table.setAttribute ("class", "msg-nested-table");
  3676.  
  3677.         td.appendChild (table);
  3678.         var tbody =  document.createElementNS ("http://www.w3.org/1999/xhtml",
  3679.                                                "html:tbody");
  3680.  
  3681.         tbody.appendChild(msgRow);
  3682.         table.appendChild(tbody);
  3683.         msgRow = tr;
  3684.     }
  3685.  
  3686.     // Actually add the item.
  3687.     addHistory (this, msgRow, canMergeData);
  3688.  
  3689.     // Update attention states...
  3690.     if (isImportant || getAttention)
  3691.     {
  3692.         setTabState(this, "attention");
  3693.         if (client.prefs["notify.aggressive"])
  3694.             window.getAttention();
  3695.     }
  3696.     else
  3697.     {
  3698.         if (isSuperfluous)
  3699.         {
  3700.             setTabState(this, "superfluous");
  3701.         }
  3702.         else
  3703.         {
  3704.             setTabState(this, "activity");
  3705.         }
  3706.     }
  3707.  
  3708.     // Copy Important Messages [to network view].
  3709.     if (isImportant && client.prefs["copyMessages"] && (o.network != this))
  3710.     {
  3711.         o.network.displayHere("{" + this.unicodeName + "} " + message, msgtype,
  3712.                               sourceObj, destObj);
  3713.     }
  3714.  
  3715.     // Log file time!
  3716.     if (this.prefs["log"])
  3717.     {
  3718.         if (!this.logFile)
  3719.             client.openLogFile(this);
  3720.  
  3721.         try
  3722.         {
  3723.             this.logFile.write(fromUnicode(logString + client.lineEnd, "utf-8"));
  3724.         }
  3725.         catch (ex)
  3726.         {
  3727.             // Stop logging before showing any messages!
  3728.             this.prefs["log"] = false;
  3729.             dd("Log file write error: " + formatException(ex));
  3730.             this.displayHere(getMsg(MSG_LOGFILE_WRITE_ERROR, getLogPath(this)),
  3731.                              "ERROR");
  3732.         }
  3733.     }
  3734. }
  3735.  
  3736. function addHistory (source, obj, mergeData)
  3737. {
  3738.     if (!("messages" in source) || (source.messages == null))
  3739.         createMessages(source);
  3740.  
  3741.     var tbody = source.messages.firstChild;
  3742.     var appendTo = tbody;
  3743.  
  3744.     var needScroll = false;
  3745.  
  3746.     if (mergeData)
  3747.     {
  3748.         var inobj = obj;
  3749.         // This gives us the non-nested row when there is nesting.
  3750.         if (inobj.className == "msg-nested-tr")
  3751.             inobj = inobj.firstChild.firstChild.firstChild.firstChild;
  3752.  
  3753.         var thisUserCol = inobj.firstChild;
  3754.         while (thisUserCol && !thisUserCol.className.match(/^(msg-user|msg-type)$/))
  3755.             thisUserCol = thisUserCol.nextSibling;
  3756.  
  3757.         var thisMessageCol = inobj.firstChild;
  3758.         while (thisMessageCol && !(thisMessageCol.className == "msg-data"))
  3759.             thisMessageCol = thisMessageCol.nextSibling;
  3760.  
  3761.         var ci = findPreviousColumnInfo(source.messages);
  3762.         var nickColumns = ci.nickColumns;
  3763.         var rowExtents = ci.extents;
  3764.         var nickColumnCount = nickColumns.length;
  3765.  
  3766.         // Are we the same user as last time?
  3767.         var sameNick = (nickColumnCount > 0 &&
  3768.                         nickColumns[nickColumnCount - 1].parentNode.
  3769.                         getAttribute("msg-user") ==
  3770.                         thisUserCol.parentNode.getAttribute("msg-user"));
  3771.  
  3772.         // What was the span last time?
  3773.         var lastRowSpan = (nickColumnCount > 0) ?
  3774.             Number(nickColumns[0].getAttribute("rowspan")) : 0;
  3775.  
  3776.         if (sameNick)
  3777.         {
  3778.             obj = inobj;
  3779.             if (ci.nested)
  3780.                 appendTo = source.messages.firstChild.lastChild.firstChild.firstChild.firstChild;
  3781.  
  3782.             if (obj.getAttribute("important"))
  3783.             {
  3784.                 nickColumns[nickColumnCount - 1].setAttribute("important",
  3785.                                                               true);
  3786.             }
  3787.  
  3788.             // Remove nickname column from new row.
  3789.             obj.removeChild(thisUserCol);
  3790.  
  3791.             // Expand previous grouping's nickname cell(s) to fill-in the gap.
  3792.             for (var i = 0; i < nickColumns.length; ++i)
  3793.                 nickColumns[i].setAttribute("rowspan", rowExtents.length + 1);
  3794.         }
  3795.     }
  3796.  
  3797.     if ("frame" in source)
  3798.         needScroll = checkScroll(source.frame);
  3799.     if (obj)
  3800.         appendTo.appendChild(obj);
  3801.  
  3802.     if (source.MAX_MESSAGES)
  3803.     {
  3804.         if (typeof source.messageCount != "number")
  3805.             source.messageCount = 1;
  3806.         else
  3807.             source.messageCount++;
  3808.  
  3809.         if (source.messageCount > source.MAX_MESSAGES)
  3810.         {
  3811.             if (client.PRINT_DIRECTION == 1)
  3812.             {
  3813.                 var height = tbody.firstChild.scrollHeight;
  3814.                 var window = getContentWindow(source.frame);
  3815.                 var x = window.pageXOffset;
  3816.                 var y = window.pageYOffset;
  3817.                 tbody.removeChild (tbody.firstChild);
  3818.                 --source.messageCount;
  3819.                 while (tbody.firstChild && tbody.firstChild.childNodes[1] &&
  3820.                        tbody.firstChild.childNodes[1].getAttribute("class") ==
  3821.                        "msg-data")
  3822.                 {
  3823.                     --source.messageCount;
  3824.                     tbody.removeChild (tbody.firstChild);
  3825.                 }
  3826.                 if (!checkScroll(source.frame) && (y > height))
  3827.                     window.scrollTo(x, y - height);
  3828.             }
  3829.             else
  3830.             {
  3831.                 tbody.removeChild (tbody.lastChild);
  3832.                 --source.messageCount;
  3833.                 while (tbody.lastChild && tbody.lastChild.childNodes[1] &&
  3834.                        tbody.lastChild.childNodes[1].getAttribute("class") ==
  3835.                        "msg-data")
  3836.                 {
  3837.                     --source.messageCount;
  3838.                     tbody.removeChild (tbody.lastChild);
  3839.                 }
  3840.             }
  3841.         }
  3842.     }
  3843.  
  3844.     if (needScroll)
  3845.     {
  3846.         scrollDown(source.frame, true);
  3847.         setTimeout(scrollDown, 500, source.frame, false);
  3848.         setTimeout(scrollDown, 1000, source.frame, false);
  3849.         setTimeout(scrollDown, 2000, source.frame, false);
  3850.     }
  3851. }
  3852.  
  3853. function findPreviousColumnInfo(table)
  3854. {
  3855.     // All the rows in the grouping (for merged rows).
  3856.     var extents = new Array();
  3857.     // Get the last row in the table.
  3858.     var tr = table.firstChild.lastChild;
  3859.     // Bail if there's no rows.
  3860.     if (!tr)
  3861.         return {extents: [], nickColumns: [], nested: false};
  3862.     // Get message type.
  3863.     if (tr.className == "msg-nested-tr")
  3864.     {
  3865.         var rv = findPreviousColumnInfo(tr.firstChild.firstChild);
  3866.         rv.nested = true;
  3867.         return rv;
  3868.     }
  3869.     // Now get the read one...
  3870.     var className = (tr && tr.childNodes[1]) ? tr.childNodes[1].getAttribute("class") : "";
  3871.     // Keep going up rows until you find the first in a group.
  3872.     // This will go up until it hits the top of a multiline/merged block.
  3873.     while (tr && tr.childNodes[1] && className.search(/msg-user|msg-type/) == -1)
  3874.     {
  3875.         extents.push(tr);
  3876.         tr = tr.previousSibling;
  3877.         if (tr && tr.childNodes[1])
  3878.             className = tr.childNodes[1].getAttribute("class");
  3879.     }
  3880.  
  3881.     // If we ran out of rows, or it's not a talking line, we're outta here.
  3882.     if (!tr || className != "msg-user")
  3883.         return {extents: [], nickColumns: [], nested: false};
  3884.  
  3885.     extents.push(tr);
  3886.  
  3887.     // Time to collect the nick data...
  3888.     var nickCol = tr.firstChild;
  3889.     // All the cells that contain nickname info.
  3890.     var nickCols = new Array();
  3891.     while (nickCol)
  3892.     {
  3893.         // Just collect nickname column cells.
  3894.         if (nickCol.getAttribute("class") == "msg-user")
  3895.             nickCols.push(nickCol);
  3896.         nickCol = nickCol.nextSibling;
  3897.     }
  3898.  
  3899.     // And we're done.
  3900.     return {extents: extents, nickColumns: nickCols, nested: false};
  3901. }
  3902.  
  3903. function getLogPath(obj)
  3904. {
  3905.     return getURLSpecFromFile(obj.prefs["logFileName"]);
  3906. }
  3907.  
  3908. client.getConnectionCount =
  3909. function cli_gccount ()
  3910. {
  3911.     var count = 0;
  3912.  
  3913.     for (var n in client.networks)
  3914.     {
  3915.         if (client.networks[n].isConnected())
  3916.             ++count;
  3917.     }
  3918.  
  3919.     return count;
  3920. }
  3921.  
  3922. client.quit =
  3923. function cli_quit (reason)
  3924. {
  3925.     for (var n in client.networks)
  3926.     {
  3927.         if (client.networks[n].isConnected())
  3928.             client.networks[n].quit(reason);
  3929.     }
  3930. }
  3931.  
  3932. /* gets a tab-complete match for the line of text specified by |line|.
  3933.  * wordStart is the position within |line| that starts the word being matched,
  3934.  * wordEnd marks the end position.  |cursorPos| marks the position of the caret
  3935.  * in the textbox.
  3936.  */
  3937. client.performTabMatch =
  3938. function gettabmatch_usr (line, wordStart, wordEnd, word, cursorPos)
  3939. {
  3940.     if (wordStart != 0 || line[0] != client.COMMAND_CHAR)
  3941.         return null;
  3942.  
  3943.     var matches = client.commandManager.listNames(word.substr(1), CMD_CONSOLE);
  3944.     if (matches.length == 1 && wordEnd == line.length)
  3945.     {
  3946.         matches[0] = client.COMMAND_CHAR + matches[0] + " ";
  3947.     }
  3948.     else
  3949.     {
  3950.         for (var i in matches)
  3951.             matches[i] = client.COMMAND_CHAR + matches[i];
  3952.     }
  3953.  
  3954.     return matches;
  3955. }
  3956.  
  3957. client.openLogFile =
  3958. function cli_startlog (view)
  3959. {
  3960.     const NORMAL_FILE_TYPE = Components.interfaces.nsIFile.NORMAL_FILE_TYPE;
  3961.  
  3962.     try
  3963.     {
  3964.         var file = new LocalFile(view.prefs["logFileName"]);
  3965.         if (!file.localFile.exists())
  3966.         {
  3967.             // futils.umask may be 0022. Result is 0644.
  3968.             file.localFile.create(NORMAL_FILE_TYPE, 0666 & ~futils.umask);
  3969.         }
  3970.         view.logFile = fopen(file.localFile, ">>");
  3971.     }
  3972.     catch (ex)
  3973.     {
  3974.         view.prefs["log"] = false;
  3975.         dd("Log file open error: " + formatException(ex));
  3976.         view.displayHere(getMsg(MSG_LOGFILE_ERROR, getLogPath(view)), MT_ERROR);
  3977.         return;
  3978.     }
  3979.  
  3980.     view.displayHere(getMsg(MSG_LOGFILE_OPENED, getLogPath(view)));
  3981. }
  3982.  
  3983. client.closeLogFile =
  3984. function cli_stoplog (view)
  3985. {
  3986.     view.displayHere(getMsg(MSG_LOGFILE_CLOSING, getLogPath(view)));
  3987.  
  3988.     if (view.logFile)
  3989.     {
  3990.         view.logFile.close();
  3991.         view.logFile = null;
  3992.     }
  3993. }
  3994.  
  3995. CIRCChannel.prototype.getLCFunction =
  3996. CIRCNetwork.prototype.getLCFunction =
  3997. CIRCUser.prototype.getLCFunction    =
  3998. CIRCDCCChat.prototype.getLCFunction =
  3999. CIRCDCCFileTransfer.prototype.getLCFunction =
  4000. function getlcfn()
  4001. {
  4002.     var details = getObjectDetails(this);
  4003.     var lcFn;
  4004.  
  4005.     if (details.server)
  4006.     {
  4007.         lcFn = function(text)
  4008.             {
  4009.                 return details.server.toLowerCase(text);
  4010.             }
  4011.     }
  4012.  
  4013.     return lcFn;
  4014. }
  4015.  
  4016. CIRCChannel.prototype.performTabMatch =
  4017. CIRCNetwork.prototype.performTabMatch =
  4018. CIRCUser.prototype.performTabMatch    =
  4019. CIRCDCCChat.prototype.performTabMatch =
  4020. CIRCDCCFileTransfer.prototype.performTabMatch =
  4021. function gettabmatch_other (line, wordStart, wordEnd, word, cursorpos, lcFn)
  4022. {
  4023.     if (wordStart == 0 && line[0] == client.COMMAND_CHAR)
  4024.     {
  4025.         return client.performTabMatch(line, wordStart, wordEnd, word,
  4026.                                       cursorpos);
  4027.     }
  4028.  
  4029.     var matchList = new Array();
  4030.     var users;
  4031.     var channels;
  4032.     var userIndex = -1;
  4033.  
  4034.     var details = getObjectDetails(this);
  4035.  
  4036.     if (details.channel && word == details.channel.unicodeName[0])
  4037.     {
  4038.         /* When we have #<tab>, we just want the current channel,
  4039.            if possible. */
  4040.         matchList.push(details.channel.unicodeName);
  4041.     }
  4042.     else
  4043.     {
  4044.         /* Ok, not #<tab> or no current channel, so get the full list. */
  4045.  
  4046.         if (details.channel)
  4047.             users = details.channel.users;
  4048.  
  4049.         if (details.server)
  4050.         {
  4051.             channels = details.server.channels;
  4052.             for (var c in channels)
  4053.                 matchList.push(channels[c].unicodeName);
  4054.             if (!users)
  4055.                 users = details.server.users;
  4056.         }
  4057.  
  4058.         if (users)
  4059.         {
  4060.             userIndex = matchList.length;
  4061.             for (var n in users)
  4062.                 matchList.push(users[n].unicodeName);
  4063.         }
  4064.     }
  4065.  
  4066.     var matches = matchEntry(word, matchList, lcFn);
  4067.  
  4068.     var list = new Array();
  4069.     for (var i = 0; i < matches.length; i++)
  4070.         list.push(matchList[matches[i]]);
  4071.  
  4072.     if (list.length == 1)
  4073.     {
  4074.         if (users && (userIndex >= 0) && (matches[0] >= userIndex))
  4075.         {
  4076.             if (wordStart == 0)
  4077.                 list[0] += client.prefs["nickCompleteStr"];
  4078.         }
  4079.  
  4080.         if (wordEnd == line.length)
  4081.         {
  4082.             /* add a space if the word is at the end of the line. */
  4083.             list[0] += " ";
  4084.         }
  4085.     }
  4086.  
  4087.     return list;
  4088. }
  4089.  
  4090. CIRCChannel.prototype.getGraphResource =
  4091. function my_graphres ()
  4092. {
  4093.     if (!("rdfRes" in this))
  4094.     {
  4095.         var id = RES_PFX + "CHANNEL:" + this.parent.parent.unicodeName + ":" +
  4096.             escape(this.unicodeName);
  4097.         this.rdfRes = client.rdf.GetResource(id);
  4098.     }
  4099.  
  4100.     return this.rdfRes;
  4101. }
  4102.  
  4103. CIRCUser.prototype.getGraphResource =
  4104. function usr_graphres()
  4105. {
  4106.     if (!ASSERT(this.TYPE == "IRCChanUser",
  4107.                 "cuser.getGraphResource called on wrong object"))
  4108.     {
  4109.         return null;
  4110.     }
  4111.  
  4112.     var rdf = client.rdf;
  4113.  
  4114.     if (!("rdfRes" in this))
  4115.     {
  4116.         if (!("nextResID" in CIRCUser))
  4117.             CIRCUser.nextResID = 0;
  4118.  
  4119.         this.rdfRes = rdf.GetResource(RES_PFX + "CUSER:" +
  4120.                                       this.parent.parent.parent.unicodeName + ":" +
  4121.                                       this.parent.unicodeName + ":" +
  4122.                                       CIRCUser.nextResID++);
  4123.  
  4124.             //dd ("created cuser resource " + this.rdfRes.Value);
  4125.  
  4126.         rdf.Assert (this.rdfRes, rdf.resNick, rdf.GetLiteral(this.unicodeName));
  4127.         rdf.Assert (this.rdfRes, rdf.resUniName, rdf.GetLiteral(this.unicodeName));
  4128.         rdf.Assert (this.rdfRes, rdf.resUser, rdf.litUnk);
  4129.         rdf.Assert (this.rdfRes, rdf.resHost, rdf.litUnk);
  4130.         rdf.Assert (this.rdfRes, rdf.resSortName, rdf.litUnk);
  4131.         rdf.Assert (this.rdfRes, rdf.resOp, rdf.litUnk);
  4132.         rdf.Assert (this.rdfRes, rdf.resHalfOp, rdf.litUnk);
  4133.         rdf.Assert (this.rdfRes, rdf.resVoice, rdf.litUnk);
  4134.         rdf.Assert (this.rdfRes, rdf.resAway, rdf.litUnk);
  4135.         this.updateGraphResource();
  4136.     }
  4137.  
  4138.     return this.rdfRes;
  4139. }
  4140.  
  4141. CIRCUser.prototype.updateGraphResource =
  4142. function usr_updres()
  4143. {
  4144.     if (!ASSERT(this.TYPE == "IRCChanUser",
  4145.                 "cuser.updateGraphResource called on wrong object"))
  4146.     {
  4147.         return;
  4148.     }
  4149.  
  4150.     if (!("rdfRes" in this))
  4151.     {
  4152.         this.getGraphResource();
  4153.         return;
  4154.     }
  4155.  
  4156.     var rdf = client.rdf;
  4157.  
  4158.     rdf.Change (this.rdfRes, rdf.resUniName, rdf.GetLiteral(this.unicodeName));
  4159.     if (this.name)
  4160.         rdf.Change (this.rdfRes, rdf.resUser, rdf.GetLiteral(this.name));
  4161.     else
  4162.         rdf.Change (this.rdfRes, rdf.resUser, rdf.litUnk);
  4163.     if (this.host)
  4164.         rdf.Change (this.rdfRes, rdf.resHost, rdf.GetLiteral(this.host));
  4165.     else
  4166.         rdf.Change (this.rdfRes, rdf.resHost, rdf.litUnk);
  4167.  
  4168.     // Check for the highest mode the user has.
  4169.     const userModes = this.parent.parent.userModes;
  4170.     var modeLevel = 0;
  4171.     var mode = "";
  4172.     for (var i = 0; i < this.modes.length; i++)
  4173.     {
  4174.         for (var j = 0; j < userModes.length; j++)
  4175.         {
  4176.             if (userModes[j].mode == this.modes[i])
  4177.             {
  4178.                 if (userModes.length - j > modeLevel)
  4179.                 {
  4180.                     modeLevel = userModes.length - j;
  4181.                     mode = userModes[j].symbol;
  4182.                 }
  4183.                 break;
  4184.             }
  4185.         }
  4186.     }
  4187.  
  4188.     // Counts up from Z to A.
  4189.     var sortname = String.fromCharCode(90 - modeLevel) + "-" + this.unicodeName;
  4190.  
  4191.     // We want to show mode symbols, but only those we don't 'style'.
  4192.     if (mode && !mode.match(/^[@%+]$/))
  4193.     {
  4194.         rdf.Change(this.rdfRes, rdf.resNick,
  4195.                    rdf.GetLiteral(mode + " " + this.unicodeName));
  4196.     }
  4197.     else
  4198.     {
  4199.         rdf.Change (this.rdfRes, rdf.resNick, rdf.GetLiteral(this.unicodeName));
  4200.     }
  4201.     rdf.Change(this.rdfRes, rdf.resSortName, rdf.GetLiteral(sortname));
  4202.     rdf.Change(this.rdfRes, rdf.resOp,
  4203.                this.isOp ? rdf.litTrue : rdf.litFalse);
  4204.     rdf.Change(this.rdfRes, rdf.resHalfOp,
  4205.                this.isHalfOp ? rdf.litTrue : rdf.litFalse);
  4206.     rdf.Change(this.rdfRes, rdf.resVoice,
  4207.                this.isVoice ? rdf.litTrue : rdf.litFalse);
  4208.     rdf.Change(this.rdfRes, rdf.resAway,
  4209.                this.isAway ? rdf.litTrue : rdf.litFalse);
  4210. }
  4211.